2005-01-07 15:47:09 +08:00
|
|
|
//===-- LegalizeDAG.cpp - Implement SelectionDAG::Legalize ----------------===//
|
2005-04-22 06:36:52 +08:00
|
|
|
//
|
2005-01-07 15:47:09 +08:00
|
|
|
// The LLVM Compiler Infrastructure
|
|
|
|
//
|
|
|
|
// This file was developed by the LLVM research group and is distributed under
|
|
|
|
// the University of Illinois Open Source License. See LICENSE.TXT for details.
|
2005-04-22 06:36:52 +08:00
|
|
|
//
|
2005-01-07 15:47:09 +08:00
|
|
|
//===----------------------------------------------------------------------===//
|
|
|
|
//
|
|
|
|
// This file implements the SelectionDAG::Legalize method.
|
|
|
|
//
|
|
|
|
//===----------------------------------------------------------------------===//
|
|
|
|
|
|
|
|
#include "llvm/CodeGen/SelectionDAG.h"
|
|
|
|
#include "llvm/CodeGen/MachineFunction.h"
|
2005-01-15 15:15:18 +08:00
|
|
|
#include "llvm/CodeGen/MachineFrameInfo.h"
|
2005-08-18 01:42:52 +08:00
|
|
|
#include "llvm/Support/MathExtras.h"
|
2005-01-07 15:47:09 +08:00
|
|
|
#include "llvm/Target/TargetLowering.h"
|
2005-01-11 13:57:22 +08:00
|
|
|
#include "llvm/Target/TargetData.h"
|
2005-01-15 14:18:18 +08:00
|
|
|
#include "llvm/Target/TargetOptions.h"
|
2005-05-14 02:50:42 +08:00
|
|
|
#include "llvm/CallingConv.h"
|
2005-01-07 15:47:09 +08:00
|
|
|
#include "llvm/Constants.h"
|
|
|
|
#include <iostream>
|
2005-08-06 02:10:27 +08:00
|
|
|
#include <set>
|
2005-01-07 15:47:09 +08:00
|
|
|
using namespace llvm;
|
|
|
|
|
|
|
|
//===----------------------------------------------------------------------===//
|
|
|
|
/// SelectionDAGLegalize - This takes an arbitrary SelectionDAG as input and
|
|
|
|
/// hacks on it until the target machine can handle it. This involves
|
|
|
|
/// eliminating value sizes the machine cannot handle (promoting small sizes to
|
|
|
|
/// large sizes or splitting up large values into small values) as well as
|
|
|
|
/// eliminating operations the machine cannot handle.
|
|
|
|
///
|
|
|
|
/// This code also does a small amount of optimization and recognition of idioms
|
|
|
|
/// as part of its processing. For example, if a target does not support a
|
|
|
|
/// 'setcc' instruction efficiently, but does support 'brcc' instruction, this
|
|
|
|
/// will attempt merge setcc and brc instructions into brcc's.
|
|
|
|
///
|
|
|
|
namespace {
|
|
|
|
class SelectionDAGLegalize {
|
|
|
|
TargetLowering &TLI;
|
|
|
|
SelectionDAG &DAG;
|
|
|
|
|
|
|
|
/// LegalizeAction - This enum indicates what action we should take for each
|
|
|
|
/// value type the can occur in the program.
|
|
|
|
enum LegalizeAction {
|
|
|
|
Legal, // The target natively supports this value type.
|
|
|
|
Promote, // This should be promoted to the next larger type.
|
|
|
|
Expand, // This integer type should be broken into smaller pieces.
|
|
|
|
};
|
|
|
|
|
|
|
|
/// ValueTypeActions - This is a bitvector that contains two bits for each
|
|
|
|
/// value type, where the two bits correspond to the LegalizeAction enum.
|
|
|
|
/// This can be queried with "getTypeAction(VT)".
|
2005-11-29 13:45:29 +08:00
|
|
|
unsigned long long ValueTypeActions;
|
2005-01-07 15:47:09 +08:00
|
|
|
|
|
|
|
/// NeedsAnotherIteration - This is set when we expand a large integer
|
|
|
|
/// operation into smaller integer operations, but the smaller operations are
|
|
|
|
/// not set. This occurs only rarely in practice, for targets that don't have
|
|
|
|
/// 32-bit or larger integer registers.
|
|
|
|
bool NeedsAnotherIteration;
|
|
|
|
|
|
|
|
/// LegalizedNodes - For nodes that are of legal width, and that have more
|
|
|
|
/// than one use, this map indicates what regularized operand to use. This
|
|
|
|
/// allows us to avoid legalizing the same thing more than once.
|
|
|
|
std::map<SDOperand, SDOperand> LegalizedNodes;
|
|
|
|
|
2005-01-15 13:21:40 +08:00
|
|
|
/// PromotedNodes - For nodes that are below legal width, and that have more
|
|
|
|
/// than one use, this map indicates what promoted value to use. This allows
|
|
|
|
/// us to avoid promoting the same thing more than once.
|
|
|
|
std::map<SDOperand, SDOperand> PromotedNodes;
|
|
|
|
|
2005-01-07 15:47:09 +08:00
|
|
|
/// ExpandedNodes - For nodes that need to be expanded, and which have more
|
|
|
|
/// than one use, this map indicates which which operands are the expanded
|
|
|
|
/// version of the input. This allows us to avoid expanding the same node
|
|
|
|
/// more than once.
|
|
|
|
std::map<SDOperand, std::pair<SDOperand, SDOperand> > ExpandedNodes;
|
|
|
|
|
2005-01-08 06:28:47 +08:00
|
|
|
void AddLegalizedOperand(SDOperand From, SDOperand To) {
|
2005-12-20 08:53:54 +08:00
|
|
|
LegalizedNodes.insert(std::make_pair(From, To));
|
|
|
|
// If someone requests legalization of the new node, return itself.
|
|
|
|
if (From != To)
|
|
|
|
LegalizedNodes.insert(std::make_pair(To, To));
|
2005-01-08 06:28:47 +08:00
|
|
|
}
|
2005-01-15 13:21:40 +08:00
|
|
|
void AddPromotedOperand(SDOperand From, SDOperand To) {
|
|
|
|
bool isNew = PromotedNodes.insert(std::make_pair(From, To)).second;
|
|
|
|
assert(isNew && "Got into the map somehow?");
|
2005-12-20 08:53:54 +08:00
|
|
|
// If someone requests legalization of the new node, return itself.
|
|
|
|
LegalizedNodes.insert(std::make_pair(To, To));
|
2005-01-15 13:21:40 +08:00
|
|
|
}
|
2005-01-08 06:28:47 +08:00
|
|
|
|
2005-01-07 15:47:09 +08:00
|
|
|
public:
|
|
|
|
|
2005-01-23 12:42:50 +08:00
|
|
|
SelectionDAGLegalize(SelectionDAG &DAG);
|
2005-01-07 15:47:09 +08:00
|
|
|
|
|
|
|
/// Run - While there is still lowering to do, perform a pass over the DAG.
|
|
|
|
/// Most regularization can be done in a single pass, but targets that require
|
|
|
|
/// large values to be split into registers multiple times (e.g. i64 -> 4x
|
|
|
|
/// i16) require iteration for these values (the first iteration will demote
|
|
|
|
/// to i32, the second will demote to i16).
|
|
|
|
void Run() {
|
|
|
|
do {
|
|
|
|
NeedsAnotherIteration = false;
|
|
|
|
LegalizeDAG();
|
|
|
|
} while (NeedsAnotherIteration);
|
|
|
|
}
|
|
|
|
|
|
|
|
/// getTypeAction - Return how we should legalize values of this type, either
|
|
|
|
/// it is already legal or we need to expand it into multiple registers of
|
|
|
|
/// smaller integer type, or we need to promote it to a larger type.
|
|
|
|
LegalizeAction getTypeAction(MVT::ValueType VT) const {
|
|
|
|
return (LegalizeAction)((ValueTypeActions >> (2*VT)) & 3);
|
|
|
|
}
|
|
|
|
|
|
|
|
/// isTypeLegal - Return true if this type is legal on this target.
|
|
|
|
///
|
|
|
|
bool isTypeLegal(MVT::ValueType VT) const {
|
|
|
|
return getTypeAction(VT) == Legal;
|
|
|
|
}
|
|
|
|
|
|
|
|
private:
|
|
|
|
void LegalizeDAG();
|
|
|
|
|
|
|
|
SDOperand LegalizeOp(SDOperand O);
|
|
|
|
void ExpandOp(SDOperand O, SDOperand &Lo, SDOperand &Hi);
|
2005-01-15 13:21:40 +08:00
|
|
|
SDOperand PromoteOp(SDOperand O);
|
2005-01-07 15:47:09 +08:00
|
|
|
|
2005-01-21 14:05:23 +08:00
|
|
|
SDOperand ExpandLibCall(const char *Name, SDNode *Node,
|
|
|
|
SDOperand &Hi);
|
|
|
|
SDOperand ExpandIntToFP(bool isSigned, MVT::ValueType DestTy,
|
|
|
|
SDOperand Source);
|
2005-07-16 08:19:57 +08:00
|
|
|
|
2005-12-23 08:16:34 +08:00
|
|
|
SDOperand ExpandBIT_CONVERT(MVT::ValueType DestVT, SDOperand SrcOp);
|
Added generic code expansion for [signed|unsigned] i32 to [f32|f64] casts in the
legalizer. PowerPC now uses this expansion instead of ISel version.
Example:
// signed integer to double conversion
double f1(signed x) {
return (double)x;
}
// unsigned integer to double conversion
double f2(unsigned x) {
return (double)x;
}
// signed integer to float conversion
float f3(signed x) {
return (float)x;
}
// unsigned integer to float conversion
float f4(unsigned x) {
return (float)x;
}
Byte Code:
internal fastcc double %_Z2f1i(int %x) {
entry:
%tmp.1 = cast int %x to double ; <double> [#uses=1]
ret double %tmp.1
}
internal fastcc double %_Z2f2j(uint %x) {
entry:
%tmp.1 = cast uint %x to double ; <double> [#uses=1]
ret double %tmp.1
}
internal fastcc float %_Z2f3i(int %x) {
entry:
%tmp.1 = cast int %x to float ; <float> [#uses=1]
ret float %tmp.1
}
internal fastcc float %_Z2f4j(uint %x) {
entry:
%tmp.1 = cast uint %x to float ; <float> [#uses=1]
ret float %tmp.1
}
internal fastcc double %_Z2g1i(int %x) {
entry:
%buffer = alloca [2 x uint] ; <[2 x uint]*> [#uses=3]
%tmp.0 = getelementptr [2 x uint]* %buffer, int 0, int 0 ; <uint*> [#uses=1]
store uint 1127219200, uint* %tmp.0
%tmp.2 = cast int %x to uint ; <uint> [#uses=1]
%tmp.3 = xor uint %tmp.2, 2147483648 ; <uint> [#uses=1]
%tmp.5 = getelementptr [2 x uint]* %buffer, int 0, int 1 ; <uint*> [#uses=1]
store uint %tmp.3, uint* %tmp.5
%tmp.9 = cast [2 x uint]* %buffer to double* ; <double*> [#uses=1]
%tmp.10 = load double* %tmp.9 ; <double> [#uses=1]
%tmp.13 = load double* cast (long* %signed_bias to double*) ; <double> [#uses=1]
%tmp.14 = sub double %tmp.10, %tmp.13 ; <double> [#uses=1]
ret double %tmp.14
}
internal fastcc double %_Z2g2j(uint %x) {
entry:
%buffer = alloca [2 x uint] ; <[2 x uint]*> [#uses=3]
%tmp.0 = getelementptr [2 x uint]* %buffer, int 0, int 0 ; <uint*> [#uses=1]
store uint 1127219200, uint* %tmp.0
%tmp.1 = getelementptr [2 x uint]* %buffer, int 0, int 1 ; <uint*> [#uses=1]
store uint %x, uint* %tmp.1
%tmp.4 = cast [2 x uint]* %buffer to double* ; <double*> [#uses=1]
%tmp.5 = load double* %tmp.4 ; <double> [#uses=1]
%tmp.8 = load double* cast (long* %unsigned_bias to double*) ; <double> [#uses=1]
%tmp.9 = sub double %tmp.5, %tmp.8 ; <double> [#uses=1]
ret double %tmp.9
}
internal fastcc float %_Z2g3i(int %x) {
entry:
%buffer = alloca [2 x uint] ; <[2 x uint]*> [#uses=3]
%tmp.0 = getelementptr [2 x uint]* %buffer, int 0, int 0 ; <uint*> [#uses=1]
store uint 1127219200, uint* %tmp.0
%tmp.2 = cast int %x to uint ; <uint> [#uses=1]
%tmp.3 = xor uint %tmp.2, 2147483648 ; <uint> [#uses=1]
%tmp.5 = getelementptr [2 x uint]* %buffer, int 0, int 1 ; <uint*> [#uses=1]
store uint %tmp.3, uint* %tmp.5
%tmp.9 = cast [2 x uint]* %buffer to double* ; <double*> [#uses=1]
%tmp.10 = load double* %tmp.9 ; <double> [#uses=1]
%tmp.13 = load double* cast (long* %signed_bias to double*) ; <double> [#uses=1]
%tmp.14 = sub double %tmp.10, %tmp.13 ; <double> [#uses=1]
%tmp.16 = cast double %tmp.14 to float ; <float> [#uses=1]
ret float %tmp.16
}
internal fastcc float %_Z2g4j(uint %x) {
entry:
%buffer = alloca [2 x uint] ; <[2 x uint]*> [#uses=3]
%tmp.0 = getelementptr [2 x uint]* %buffer, int 0, int 0 ; <uint*> [#uses=1]
store uint 1127219200, uint* %tmp.0
%tmp.1 = getelementptr [2 x uint]* %buffer, int 0, int 1 ; <uint*> [#uses=1]
store uint %x, uint* %tmp.1
%tmp.4 = cast [2 x uint]* %buffer to double* ; <double*> [#uses=1]
%tmp.5 = load double* %tmp.4 ; <double> [#uses=1]
%tmp.8 = load double* cast (long* %unsigned_bias to double*) ; <double> [#uses=1]
%tmp.9 = sub double %tmp.5, %tmp.8 ; <double> [#uses=1]
%tmp.11 = cast double %tmp.9 to float ; <float> [#uses=1]
ret float %tmp.11
}
PowerPC Code:
.machine ppc970
.const
.align 2
.CPIl1__Z2f1i_0: ; float 0x4330000080000000
.long 1501560836 ; float 4.5036e+15
.text
.align 2
.globl l1__Z2f1i
l1__Z2f1i:
.LBBl1__Z2f1i_0: ; entry
xoris r2, r3, 32768
stw r2, -4(r1)
lis r2, 17200
stw r2, -8(r1)
lfd f0, -8(r1)
lis r2, ha16(.CPIl1__Z2f1i_0)
lfs f1, lo16(.CPIl1__Z2f1i_0)(r2)
fsub f1, f0, f1
blr
.const
.align 2
.CPIl2__Z2f2j_0: ; float 0x4330000000000000
.long 1501560832 ; float 4.5036e+15
.text
.align 2
.globl l2__Z2f2j
l2__Z2f2j:
.LBBl2__Z2f2j_0: ; entry
stw r3, -4(r1)
lis r2, 17200
stw r2, -8(r1)
lfd f0, -8(r1)
lis r2, ha16(.CPIl2__Z2f2j_0)
lfs f1, lo16(.CPIl2__Z2f2j_0)(r2)
fsub f1, f0, f1
blr
.const
.align 2
.CPIl3__Z2f3i_0: ; float 0x4330000080000000
.long 1501560836 ; float 4.5036e+15
.text
.align 2
.globl l3__Z2f3i
l3__Z2f3i:
.LBBl3__Z2f3i_0: ; entry
xoris r2, r3, 32768
stw r2, -4(r1)
lis r2, 17200
stw r2, -8(r1)
lfd f0, -8(r1)
lis r2, ha16(.CPIl3__Z2f3i_0)
lfs f1, lo16(.CPIl3__Z2f3i_0)(r2)
fsub f0, f0, f1
frsp f1, f0
blr
.const
.align 2
.CPIl4__Z2f4j_0: ; float 0x4330000000000000
.long 1501560832 ; float 4.5036e+15
.text
.align 2
.globl l4__Z2f4j
l4__Z2f4j:
.LBBl4__Z2f4j_0: ; entry
stw r3, -4(r1)
lis r2, 17200
stw r2, -8(r1)
lfd f0, -8(r1)
lis r2, ha16(.CPIl4__Z2f4j_0)
lfs f1, lo16(.CPIl4__Z2f4j_0)(r2)
fsub f0, f0, f1
frsp f1, f0
blr
llvm-svn: 22814
2005-08-17 08:39:29 +08:00
|
|
|
SDOperand ExpandLegalINT_TO_FP(bool isSigned,
|
|
|
|
SDOperand LegalOp,
|
|
|
|
MVT::ValueType DestVT);
|
Teach the legalizer how to promote SINT_TO_FP to a wider SINT_TO_FP that
the target natively supports. This eliminates some special-case code from
the x86 backend and generates better code as well.
For an i8 to f64 conversion, before & after:
_x87 before:
subl $2, %esp
movb 6(%esp), %al
movsbw %al, %ax
movw %ax, (%esp)
filds (%esp)
addl $2, %esp
ret
_x87 after:
subl $2, %esp
movsbw 6(%esp), %ax
movw %ax, (%esp)
filds (%esp)
addl $2, %esp
ret
_sse before:
subl $12, %esp
movb 16(%esp), %al
movsbl %al, %eax
cvtsi2sd %eax, %xmm0
addl $12, %esp
ret
_sse after:
subl $12, %esp
movsbl 16(%esp), %eax
cvtsi2sd %eax, %xmm0
addl $12, %esp
ret
llvm-svn: 22452
2005-07-16 10:02:34 +08:00
|
|
|
SDOperand PromoteLegalINT_TO_FP(SDOperand LegalOp, MVT::ValueType DestVT,
|
|
|
|
bool isSigned);
|
2005-07-29 08:11:56 +08:00
|
|
|
SDOperand PromoteLegalFP_TO_INT(SDOperand LegalOp, MVT::ValueType DestVT,
|
|
|
|
bool isSigned);
|
2005-07-27 14:12:32 +08:00
|
|
|
|
2005-01-19 12:19:40 +08:00
|
|
|
bool ExpandShift(unsigned Opc, SDOperand Op, SDOperand Amt,
|
|
|
|
SDOperand &Lo, SDOperand &Hi);
|
2005-04-02 12:00:59 +08:00
|
|
|
void ExpandShiftParts(unsigned NodeOp, SDOperand Op, SDOperand Amt,
|
|
|
|
SDOperand &Lo, SDOperand &Hi);
|
|
|
|
void ExpandByParts(unsigned NodeOp, SDOperand LHS, SDOperand RHS,
|
2005-04-02 11:38:53 +08:00
|
|
|
SDOperand &Lo, SDOperand &Hi);
|
2005-01-19 12:19:40 +08:00
|
|
|
|
2005-05-12 12:49:08 +08:00
|
|
|
void SpliceCallInto(const SDOperand &CallResult, SDNode *OutChain);
|
|
|
|
|
2005-01-07 15:47:09 +08:00
|
|
|
SDOperand getIntPtrConstant(uint64_t Val) {
|
|
|
|
return DAG.getConstant(Val, TLI.getPointerTy());
|
|
|
|
}
|
|
|
|
};
|
|
|
|
}
|
|
|
|
|
2005-11-19 13:51:46 +08:00
|
|
|
static unsigned getScalarizedOpcode(unsigned VecOp, MVT::ValueType VT) {
|
2005-11-19 08:36:38 +08:00
|
|
|
switch (VecOp) {
|
|
|
|
default: assert(0 && "Don't know how to scalarize this opcode!");
|
|
|
|
case ISD::VADD: return MVT::isInteger(VT) ? ISD::ADD : ISD::FADD;
|
|
|
|
case ISD::VSUB: return MVT::isInteger(VT) ? ISD::SUB : ISD::FSUB;
|
|
|
|
case ISD::VMUL: return MVT::isInteger(VT) ? ISD::MUL : ISD::FMUL;
|
|
|
|
}
|
|
|
|
}
|
2005-01-07 15:47:09 +08:00
|
|
|
|
2005-01-23 12:42:50 +08:00
|
|
|
SelectionDAGLegalize::SelectionDAGLegalize(SelectionDAG &dag)
|
|
|
|
: TLI(dag.getTargetLoweringInfo()), DAG(dag),
|
|
|
|
ValueTypeActions(TLI.getValueTypeActions()) {
|
2005-11-29 13:45:29 +08:00
|
|
|
assert(MVT::LAST_VALUETYPE <= 32 &&
|
2005-01-07 15:47:09 +08:00
|
|
|
"Too many value types for ValueTypeActions to hold!");
|
|
|
|
}
|
|
|
|
|
Added generic code expansion for [signed|unsigned] i32 to [f32|f64] casts in the
legalizer. PowerPC now uses this expansion instead of ISel version.
Example:
// signed integer to double conversion
double f1(signed x) {
return (double)x;
}
// unsigned integer to double conversion
double f2(unsigned x) {
return (double)x;
}
// signed integer to float conversion
float f3(signed x) {
return (float)x;
}
// unsigned integer to float conversion
float f4(unsigned x) {
return (float)x;
}
Byte Code:
internal fastcc double %_Z2f1i(int %x) {
entry:
%tmp.1 = cast int %x to double ; <double> [#uses=1]
ret double %tmp.1
}
internal fastcc double %_Z2f2j(uint %x) {
entry:
%tmp.1 = cast uint %x to double ; <double> [#uses=1]
ret double %tmp.1
}
internal fastcc float %_Z2f3i(int %x) {
entry:
%tmp.1 = cast int %x to float ; <float> [#uses=1]
ret float %tmp.1
}
internal fastcc float %_Z2f4j(uint %x) {
entry:
%tmp.1 = cast uint %x to float ; <float> [#uses=1]
ret float %tmp.1
}
internal fastcc double %_Z2g1i(int %x) {
entry:
%buffer = alloca [2 x uint] ; <[2 x uint]*> [#uses=3]
%tmp.0 = getelementptr [2 x uint]* %buffer, int 0, int 0 ; <uint*> [#uses=1]
store uint 1127219200, uint* %tmp.0
%tmp.2 = cast int %x to uint ; <uint> [#uses=1]
%tmp.3 = xor uint %tmp.2, 2147483648 ; <uint> [#uses=1]
%tmp.5 = getelementptr [2 x uint]* %buffer, int 0, int 1 ; <uint*> [#uses=1]
store uint %tmp.3, uint* %tmp.5
%tmp.9 = cast [2 x uint]* %buffer to double* ; <double*> [#uses=1]
%tmp.10 = load double* %tmp.9 ; <double> [#uses=1]
%tmp.13 = load double* cast (long* %signed_bias to double*) ; <double> [#uses=1]
%tmp.14 = sub double %tmp.10, %tmp.13 ; <double> [#uses=1]
ret double %tmp.14
}
internal fastcc double %_Z2g2j(uint %x) {
entry:
%buffer = alloca [2 x uint] ; <[2 x uint]*> [#uses=3]
%tmp.0 = getelementptr [2 x uint]* %buffer, int 0, int 0 ; <uint*> [#uses=1]
store uint 1127219200, uint* %tmp.0
%tmp.1 = getelementptr [2 x uint]* %buffer, int 0, int 1 ; <uint*> [#uses=1]
store uint %x, uint* %tmp.1
%tmp.4 = cast [2 x uint]* %buffer to double* ; <double*> [#uses=1]
%tmp.5 = load double* %tmp.4 ; <double> [#uses=1]
%tmp.8 = load double* cast (long* %unsigned_bias to double*) ; <double> [#uses=1]
%tmp.9 = sub double %tmp.5, %tmp.8 ; <double> [#uses=1]
ret double %tmp.9
}
internal fastcc float %_Z2g3i(int %x) {
entry:
%buffer = alloca [2 x uint] ; <[2 x uint]*> [#uses=3]
%tmp.0 = getelementptr [2 x uint]* %buffer, int 0, int 0 ; <uint*> [#uses=1]
store uint 1127219200, uint* %tmp.0
%tmp.2 = cast int %x to uint ; <uint> [#uses=1]
%tmp.3 = xor uint %tmp.2, 2147483648 ; <uint> [#uses=1]
%tmp.5 = getelementptr [2 x uint]* %buffer, int 0, int 1 ; <uint*> [#uses=1]
store uint %tmp.3, uint* %tmp.5
%tmp.9 = cast [2 x uint]* %buffer to double* ; <double*> [#uses=1]
%tmp.10 = load double* %tmp.9 ; <double> [#uses=1]
%tmp.13 = load double* cast (long* %signed_bias to double*) ; <double> [#uses=1]
%tmp.14 = sub double %tmp.10, %tmp.13 ; <double> [#uses=1]
%tmp.16 = cast double %tmp.14 to float ; <float> [#uses=1]
ret float %tmp.16
}
internal fastcc float %_Z2g4j(uint %x) {
entry:
%buffer = alloca [2 x uint] ; <[2 x uint]*> [#uses=3]
%tmp.0 = getelementptr [2 x uint]* %buffer, int 0, int 0 ; <uint*> [#uses=1]
store uint 1127219200, uint* %tmp.0
%tmp.1 = getelementptr [2 x uint]* %buffer, int 0, int 1 ; <uint*> [#uses=1]
store uint %x, uint* %tmp.1
%tmp.4 = cast [2 x uint]* %buffer to double* ; <double*> [#uses=1]
%tmp.5 = load double* %tmp.4 ; <double> [#uses=1]
%tmp.8 = load double* cast (long* %unsigned_bias to double*) ; <double> [#uses=1]
%tmp.9 = sub double %tmp.5, %tmp.8 ; <double> [#uses=1]
%tmp.11 = cast double %tmp.9 to float ; <float> [#uses=1]
ret float %tmp.11
}
PowerPC Code:
.machine ppc970
.const
.align 2
.CPIl1__Z2f1i_0: ; float 0x4330000080000000
.long 1501560836 ; float 4.5036e+15
.text
.align 2
.globl l1__Z2f1i
l1__Z2f1i:
.LBBl1__Z2f1i_0: ; entry
xoris r2, r3, 32768
stw r2, -4(r1)
lis r2, 17200
stw r2, -8(r1)
lfd f0, -8(r1)
lis r2, ha16(.CPIl1__Z2f1i_0)
lfs f1, lo16(.CPIl1__Z2f1i_0)(r2)
fsub f1, f0, f1
blr
.const
.align 2
.CPIl2__Z2f2j_0: ; float 0x4330000000000000
.long 1501560832 ; float 4.5036e+15
.text
.align 2
.globl l2__Z2f2j
l2__Z2f2j:
.LBBl2__Z2f2j_0: ; entry
stw r3, -4(r1)
lis r2, 17200
stw r2, -8(r1)
lfd f0, -8(r1)
lis r2, ha16(.CPIl2__Z2f2j_0)
lfs f1, lo16(.CPIl2__Z2f2j_0)(r2)
fsub f1, f0, f1
blr
.const
.align 2
.CPIl3__Z2f3i_0: ; float 0x4330000080000000
.long 1501560836 ; float 4.5036e+15
.text
.align 2
.globl l3__Z2f3i
l3__Z2f3i:
.LBBl3__Z2f3i_0: ; entry
xoris r2, r3, 32768
stw r2, -4(r1)
lis r2, 17200
stw r2, -8(r1)
lfd f0, -8(r1)
lis r2, ha16(.CPIl3__Z2f3i_0)
lfs f1, lo16(.CPIl3__Z2f3i_0)(r2)
fsub f0, f0, f1
frsp f1, f0
blr
.const
.align 2
.CPIl4__Z2f4j_0: ; float 0x4330000000000000
.long 1501560832 ; float 4.5036e+15
.text
.align 2
.globl l4__Z2f4j
l4__Z2f4j:
.LBBl4__Z2f4j_0: ; entry
stw r3, -4(r1)
lis r2, 17200
stw r2, -8(r1)
lfd f0, -8(r1)
lis r2, ha16(.CPIl4__Z2f4j_0)
lfs f1, lo16(.CPIl4__Z2f4j_0)(r2)
fsub f0, f0, f1
frsp f1, f0
blr
llvm-svn: 22814
2005-08-17 08:39:29 +08:00
|
|
|
/// ExpandLegalINT_TO_FP - This function is responsible for legalizing a
|
|
|
|
/// INT_TO_FP operation of the specified operand when the target requests that
|
2005-07-16 08:19:57 +08:00
|
|
|
/// we expand it. At this point, we know that the result and operand types are
|
|
|
|
/// legal for the target.
|
Added generic code expansion for [signed|unsigned] i32 to [f32|f64] casts in the
legalizer. PowerPC now uses this expansion instead of ISel version.
Example:
// signed integer to double conversion
double f1(signed x) {
return (double)x;
}
// unsigned integer to double conversion
double f2(unsigned x) {
return (double)x;
}
// signed integer to float conversion
float f3(signed x) {
return (float)x;
}
// unsigned integer to float conversion
float f4(unsigned x) {
return (float)x;
}
Byte Code:
internal fastcc double %_Z2f1i(int %x) {
entry:
%tmp.1 = cast int %x to double ; <double> [#uses=1]
ret double %tmp.1
}
internal fastcc double %_Z2f2j(uint %x) {
entry:
%tmp.1 = cast uint %x to double ; <double> [#uses=1]
ret double %tmp.1
}
internal fastcc float %_Z2f3i(int %x) {
entry:
%tmp.1 = cast int %x to float ; <float> [#uses=1]
ret float %tmp.1
}
internal fastcc float %_Z2f4j(uint %x) {
entry:
%tmp.1 = cast uint %x to float ; <float> [#uses=1]
ret float %tmp.1
}
internal fastcc double %_Z2g1i(int %x) {
entry:
%buffer = alloca [2 x uint] ; <[2 x uint]*> [#uses=3]
%tmp.0 = getelementptr [2 x uint]* %buffer, int 0, int 0 ; <uint*> [#uses=1]
store uint 1127219200, uint* %tmp.0
%tmp.2 = cast int %x to uint ; <uint> [#uses=1]
%tmp.3 = xor uint %tmp.2, 2147483648 ; <uint> [#uses=1]
%tmp.5 = getelementptr [2 x uint]* %buffer, int 0, int 1 ; <uint*> [#uses=1]
store uint %tmp.3, uint* %tmp.5
%tmp.9 = cast [2 x uint]* %buffer to double* ; <double*> [#uses=1]
%tmp.10 = load double* %tmp.9 ; <double> [#uses=1]
%tmp.13 = load double* cast (long* %signed_bias to double*) ; <double> [#uses=1]
%tmp.14 = sub double %tmp.10, %tmp.13 ; <double> [#uses=1]
ret double %tmp.14
}
internal fastcc double %_Z2g2j(uint %x) {
entry:
%buffer = alloca [2 x uint] ; <[2 x uint]*> [#uses=3]
%tmp.0 = getelementptr [2 x uint]* %buffer, int 0, int 0 ; <uint*> [#uses=1]
store uint 1127219200, uint* %tmp.0
%tmp.1 = getelementptr [2 x uint]* %buffer, int 0, int 1 ; <uint*> [#uses=1]
store uint %x, uint* %tmp.1
%tmp.4 = cast [2 x uint]* %buffer to double* ; <double*> [#uses=1]
%tmp.5 = load double* %tmp.4 ; <double> [#uses=1]
%tmp.8 = load double* cast (long* %unsigned_bias to double*) ; <double> [#uses=1]
%tmp.9 = sub double %tmp.5, %tmp.8 ; <double> [#uses=1]
ret double %tmp.9
}
internal fastcc float %_Z2g3i(int %x) {
entry:
%buffer = alloca [2 x uint] ; <[2 x uint]*> [#uses=3]
%tmp.0 = getelementptr [2 x uint]* %buffer, int 0, int 0 ; <uint*> [#uses=1]
store uint 1127219200, uint* %tmp.0
%tmp.2 = cast int %x to uint ; <uint> [#uses=1]
%tmp.3 = xor uint %tmp.2, 2147483648 ; <uint> [#uses=1]
%tmp.5 = getelementptr [2 x uint]* %buffer, int 0, int 1 ; <uint*> [#uses=1]
store uint %tmp.3, uint* %tmp.5
%tmp.9 = cast [2 x uint]* %buffer to double* ; <double*> [#uses=1]
%tmp.10 = load double* %tmp.9 ; <double> [#uses=1]
%tmp.13 = load double* cast (long* %signed_bias to double*) ; <double> [#uses=1]
%tmp.14 = sub double %tmp.10, %tmp.13 ; <double> [#uses=1]
%tmp.16 = cast double %tmp.14 to float ; <float> [#uses=1]
ret float %tmp.16
}
internal fastcc float %_Z2g4j(uint %x) {
entry:
%buffer = alloca [2 x uint] ; <[2 x uint]*> [#uses=3]
%tmp.0 = getelementptr [2 x uint]* %buffer, int 0, int 0 ; <uint*> [#uses=1]
store uint 1127219200, uint* %tmp.0
%tmp.1 = getelementptr [2 x uint]* %buffer, int 0, int 1 ; <uint*> [#uses=1]
store uint %x, uint* %tmp.1
%tmp.4 = cast [2 x uint]* %buffer to double* ; <double*> [#uses=1]
%tmp.5 = load double* %tmp.4 ; <double> [#uses=1]
%tmp.8 = load double* cast (long* %unsigned_bias to double*) ; <double> [#uses=1]
%tmp.9 = sub double %tmp.5, %tmp.8 ; <double> [#uses=1]
%tmp.11 = cast double %tmp.9 to float ; <float> [#uses=1]
ret float %tmp.11
}
PowerPC Code:
.machine ppc970
.const
.align 2
.CPIl1__Z2f1i_0: ; float 0x4330000080000000
.long 1501560836 ; float 4.5036e+15
.text
.align 2
.globl l1__Z2f1i
l1__Z2f1i:
.LBBl1__Z2f1i_0: ; entry
xoris r2, r3, 32768
stw r2, -4(r1)
lis r2, 17200
stw r2, -8(r1)
lfd f0, -8(r1)
lis r2, ha16(.CPIl1__Z2f1i_0)
lfs f1, lo16(.CPIl1__Z2f1i_0)(r2)
fsub f1, f0, f1
blr
.const
.align 2
.CPIl2__Z2f2j_0: ; float 0x4330000000000000
.long 1501560832 ; float 4.5036e+15
.text
.align 2
.globl l2__Z2f2j
l2__Z2f2j:
.LBBl2__Z2f2j_0: ; entry
stw r3, -4(r1)
lis r2, 17200
stw r2, -8(r1)
lfd f0, -8(r1)
lis r2, ha16(.CPIl2__Z2f2j_0)
lfs f1, lo16(.CPIl2__Z2f2j_0)(r2)
fsub f1, f0, f1
blr
.const
.align 2
.CPIl3__Z2f3i_0: ; float 0x4330000080000000
.long 1501560836 ; float 4.5036e+15
.text
.align 2
.globl l3__Z2f3i
l3__Z2f3i:
.LBBl3__Z2f3i_0: ; entry
xoris r2, r3, 32768
stw r2, -4(r1)
lis r2, 17200
stw r2, -8(r1)
lfd f0, -8(r1)
lis r2, ha16(.CPIl3__Z2f3i_0)
lfs f1, lo16(.CPIl3__Z2f3i_0)(r2)
fsub f0, f0, f1
frsp f1, f0
blr
.const
.align 2
.CPIl4__Z2f4j_0: ; float 0x4330000000000000
.long 1501560832 ; float 4.5036e+15
.text
.align 2
.globl l4__Z2f4j
l4__Z2f4j:
.LBBl4__Z2f4j_0: ; entry
stw r3, -4(r1)
lis r2, 17200
stw r2, -8(r1)
lfd f0, -8(r1)
lis r2, ha16(.CPIl4__Z2f4j_0)
lfs f1, lo16(.CPIl4__Z2f4j_0)(r2)
fsub f0, f0, f1
frsp f1, f0
blr
llvm-svn: 22814
2005-08-17 08:39:29 +08:00
|
|
|
SDOperand SelectionDAGLegalize::ExpandLegalINT_TO_FP(bool isSigned,
|
|
|
|
SDOperand Op0,
|
|
|
|
MVT::ValueType DestVT) {
|
|
|
|
if (Op0.getValueType() == MVT::i32) {
|
|
|
|
// simple 32-bit [signed|unsigned] integer to float/double expansion
|
|
|
|
|
|
|
|
// get the stack frame index of a 8 byte buffer
|
|
|
|
MachineFunction &MF = DAG.getMachineFunction();
|
|
|
|
int SSFI = MF.getFrameInfo()->CreateStackObject(8, 8);
|
|
|
|
// get address of 8 byte buffer
|
|
|
|
SDOperand StackSlot = DAG.getFrameIndex(SSFI, TLI.getPointerTy());
|
|
|
|
// word offset constant for Hi/Lo address computation
|
|
|
|
SDOperand WordOff = DAG.getConstant(sizeof(int), TLI.getPointerTy());
|
|
|
|
// set up Hi and Lo (into buffer) address based on endian
|
|
|
|
SDOperand Hi, Lo;
|
|
|
|
if (TLI.isLittleEndian()) {
|
|
|
|
Hi = DAG.getNode(ISD::ADD, TLI.getPointerTy(), StackSlot, WordOff);
|
|
|
|
Lo = StackSlot;
|
|
|
|
} else {
|
|
|
|
Hi = StackSlot;
|
|
|
|
Lo = DAG.getNode(ISD::ADD, TLI.getPointerTy(), StackSlot, WordOff);
|
|
|
|
}
|
|
|
|
// if signed map to unsigned space
|
|
|
|
SDOperand Op0Mapped;
|
|
|
|
if (isSigned) {
|
|
|
|
// constant used to invert sign bit (signed to unsigned mapping)
|
|
|
|
SDOperand SignBit = DAG.getConstant(0x80000000u, MVT::i32);
|
|
|
|
Op0Mapped = DAG.getNode(ISD::XOR, MVT::i32, Op0, SignBit);
|
|
|
|
} else {
|
|
|
|
Op0Mapped = Op0;
|
|
|
|
}
|
|
|
|
// store the lo of the constructed double - based on integer input
|
|
|
|
SDOperand Store1 = DAG.getNode(ISD::STORE, MVT::Other, DAG.getEntryNode(),
|
|
|
|
Op0Mapped, Lo, DAG.getSrcValue(NULL));
|
|
|
|
// initial hi portion of constructed double
|
|
|
|
SDOperand InitialHi = DAG.getConstant(0x43300000u, MVT::i32);
|
|
|
|
// store the hi of the constructed double - biased exponent
|
|
|
|
SDOperand Store2 = DAG.getNode(ISD::STORE, MVT::Other, Store1,
|
|
|
|
InitialHi, Hi, DAG.getSrcValue(NULL));
|
|
|
|
// load the constructed double
|
|
|
|
SDOperand Load = DAG.getLoad(MVT::f64, Store2, StackSlot,
|
|
|
|
DAG.getSrcValue(NULL));
|
|
|
|
// FP constant to bias correct the final result
|
2005-08-18 01:42:52 +08:00
|
|
|
SDOperand Bias = DAG.getConstantFP(isSigned ?
|
|
|
|
BitsToDouble(0x4330000080000000ULL)
|
|
|
|
: BitsToDouble(0x4330000000000000ULL),
|
Added generic code expansion for [signed|unsigned] i32 to [f32|f64] casts in the
legalizer. PowerPC now uses this expansion instead of ISel version.
Example:
// signed integer to double conversion
double f1(signed x) {
return (double)x;
}
// unsigned integer to double conversion
double f2(unsigned x) {
return (double)x;
}
// signed integer to float conversion
float f3(signed x) {
return (float)x;
}
// unsigned integer to float conversion
float f4(unsigned x) {
return (float)x;
}
Byte Code:
internal fastcc double %_Z2f1i(int %x) {
entry:
%tmp.1 = cast int %x to double ; <double> [#uses=1]
ret double %tmp.1
}
internal fastcc double %_Z2f2j(uint %x) {
entry:
%tmp.1 = cast uint %x to double ; <double> [#uses=1]
ret double %tmp.1
}
internal fastcc float %_Z2f3i(int %x) {
entry:
%tmp.1 = cast int %x to float ; <float> [#uses=1]
ret float %tmp.1
}
internal fastcc float %_Z2f4j(uint %x) {
entry:
%tmp.1 = cast uint %x to float ; <float> [#uses=1]
ret float %tmp.1
}
internal fastcc double %_Z2g1i(int %x) {
entry:
%buffer = alloca [2 x uint] ; <[2 x uint]*> [#uses=3]
%tmp.0 = getelementptr [2 x uint]* %buffer, int 0, int 0 ; <uint*> [#uses=1]
store uint 1127219200, uint* %tmp.0
%tmp.2 = cast int %x to uint ; <uint> [#uses=1]
%tmp.3 = xor uint %tmp.2, 2147483648 ; <uint> [#uses=1]
%tmp.5 = getelementptr [2 x uint]* %buffer, int 0, int 1 ; <uint*> [#uses=1]
store uint %tmp.3, uint* %tmp.5
%tmp.9 = cast [2 x uint]* %buffer to double* ; <double*> [#uses=1]
%tmp.10 = load double* %tmp.9 ; <double> [#uses=1]
%tmp.13 = load double* cast (long* %signed_bias to double*) ; <double> [#uses=1]
%tmp.14 = sub double %tmp.10, %tmp.13 ; <double> [#uses=1]
ret double %tmp.14
}
internal fastcc double %_Z2g2j(uint %x) {
entry:
%buffer = alloca [2 x uint] ; <[2 x uint]*> [#uses=3]
%tmp.0 = getelementptr [2 x uint]* %buffer, int 0, int 0 ; <uint*> [#uses=1]
store uint 1127219200, uint* %tmp.0
%tmp.1 = getelementptr [2 x uint]* %buffer, int 0, int 1 ; <uint*> [#uses=1]
store uint %x, uint* %tmp.1
%tmp.4 = cast [2 x uint]* %buffer to double* ; <double*> [#uses=1]
%tmp.5 = load double* %tmp.4 ; <double> [#uses=1]
%tmp.8 = load double* cast (long* %unsigned_bias to double*) ; <double> [#uses=1]
%tmp.9 = sub double %tmp.5, %tmp.8 ; <double> [#uses=1]
ret double %tmp.9
}
internal fastcc float %_Z2g3i(int %x) {
entry:
%buffer = alloca [2 x uint] ; <[2 x uint]*> [#uses=3]
%tmp.0 = getelementptr [2 x uint]* %buffer, int 0, int 0 ; <uint*> [#uses=1]
store uint 1127219200, uint* %tmp.0
%tmp.2 = cast int %x to uint ; <uint> [#uses=1]
%tmp.3 = xor uint %tmp.2, 2147483648 ; <uint> [#uses=1]
%tmp.5 = getelementptr [2 x uint]* %buffer, int 0, int 1 ; <uint*> [#uses=1]
store uint %tmp.3, uint* %tmp.5
%tmp.9 = cast [2 x uint]* %buffer to double* ; <double*> [#uses=1]
%tmp.10 = load double* %tmp.9 ; <double> [#uses=1]
%tmp.13 = load double* cast (long* %signed_bias to double*) ; <double> [#uses=1]
%tmp.14 = sub double %tmp.10, %tmp.13 ; <double> [#uses=1]
%tmp.16 = cast double %tmp.14 to float ; <float> [#uses=1]
ret float %tmp.16
}
internal fastcc float %_Z2g4j(uint %x) {
entry:
%buffer = alloca [2 x uint] ; <[2 x uint]*> [#uses=3]
%tmp.0 = getelementptr [2 x uint]* %buffer, int 0, int 0 ; <uint*> [#uses=1]
store uint 1127219200, uint* %tmp.0
%tmp.1 = getelementptr [2 x uint]* %buffer, int 0, int 1 ; <uint*> [#uses=1]
store uint %x, uint* %tmp.1
%tmp.4 = cast [2 x uint]* %buffer to double* ; <double*> [#uses=1]
%tmp.5 = load double* %tmp.4 ; <double> [#uses=1]
%tmp.8 = load double* cast (long* %unsigned_bias to double*) ; <double> [#uses=1]
%tmp.9 = sub double %tmp.5, %tmp.8 ; <double> [#uses=1]
%tmp.11 = cast double %tmp.9 to float ; <float> [#uses=1]
ret float %tmp.11
}
PowerPC Code:
.machine ppc970
.const
.align 2
.CPIl1__Z2f1i_0: ; float 0x4330000080000000
.long 1501560836 ; float 4.5036e+15
.text
.align 2
.globl l1__Z2f1i
l1__Z2f1i:
.LBBl1__Z2f1i_0: ; entry
xoris r2, r3, 32768
stw r2, -4(r1)
lis r2, 17200
stw r2, -8(r1)
lfd f0, -8(r1)
lis r2, ha16(.CPIl1__Z2f1i_0)
lfs f1, lo16(.CPIl1__Z2f1i_0)(r2)
fsub f1, f0, f1
blr
.const
.align 2
.CPIl2__Z2f2j_0: ; float 0x4330000000000000
.long 1501560832 ; float 4.5036e+15
.text
.align 2
.globl l2__Z2f2j
l2__Z2f2j:
.LBBl2__Z2f2j_0: ; entry
stw r3, -4(r1)
lis r2, 17200
stw r2, -8(r1)
lfd f0, -8(r1)
lis r2, ha16(.CPIl2__Z2f2j_0)
lfs f1, lo16(.CPIl2__Z2f2j_0)(r2)
fsub f1, f0, f1
blr
.const
.align 2
.CPIl3__Z2f3i_0: ; float 0x4330000080000000
.long 1501560836 ; float 4.5036e+15
.text
.align 2
.globl l3__Z2f3i
l3__Z2f3i:
.LBBl3__Z2f3i_0: ; entry
xoris r2, r3, 32768
stw r2, -4(r1)
lis r2, 17200
stw r2, -8(r1)
lfd f0, -8(r1)
lis r2, ha16(.CPIl3__Z2f3i_0)
lfs f1, lo16(.CPIl3__Z2f3i_0)(r2)
fsub f0, f0, f1
frsp f1, f0
blr
.const
.align 2
.CPIl4__Z2f4j_0: ; float 0x4330000000000000
.long 1501560832 ; float 4.5036e+15
.text
.align 2
.globl l4__Z2f4j
l4__Z2f4j:
.LBBl4__Z2f4j_0: ; entry
stw r3, -4(r1)
lis r2, 17200
stw r2, -8(r1)
lfd f0, -8(r1)
lis r2, ha16(.CPIl4__Z2f4j_0)
lfs f1, lo16(.CPIl4__Z2f4j_0)(r2)
fsub f0, f0, f1
frsp f1, f0
blr
llvm-svn: 22814
2005-08-17 08:39:29 +08:00
|
|
|
MVT::f64);
|
|
|
|
// subtract the bias
|
2005-09-29 06:28:18 +08:00
|
|
|
SDOperand Sub = DAG.getNode(ISD::FSUB, MVT::f64, Load, Bias);
|
Added generic code expansion for [signed|unsigned] i32 to [f32|f64] casts in the
legalizer. PowerPC now uses this expansion instead of ISel version.
Example:
// signed integer to double conversion
double f1(signed x) {
return (double)x;
}
// unsigned integer to double conversion
double f2(unsigned x) {
return (double)x;
}
// signed integer to float conversion
float f3(signed x) {
return (float)x;
}
// unsigned integer to float conversion
float f4(unsigned x) {
return (float)x;
}
Byte Code:
internal fastcc double %_Z2f1i(int %x) {
entry:
%tmp.1 = cast int %x to double ; <double> [#uses=1]
ret double %tmp.1
}
internal fastcc double %_Z2f2j(uint %x) {
entry:
%tmp.1 = cast uint %x to double ; <double> [#uses=1]
ret double %tmp.1
}
internal fastcc float %_Z2f3i(int %x) {
entry:
%tmp.1 = cast int %x to float ; <float> [#uses=1]
ret float %tmp.1
}
internal fastcc float %_Z2f4j(uint %x) {
entry:
%tmp.1 = cast uint %x to float ; <float> [#uses=1]
ret float %tmp.1
}
internal fastcc double %_Z2g1i(int %x) {
entry:
%buffer = alloca [2 x uint] ; <[2 x uint]*> [#uses=3]
%tmp.0 = getelementptr [2 x uint]* %buffer, int 0, int 0 ; <uint*> [#uses=1]
store uint 1127219200, uint* %tmp.0
%tmp.2 = cast int %x to uint ; <uint> [#uses=1]
%tmp.3 = xor uint %tmp.2, 2147483648 ; <uint> [#uses=1]
%tmp.5 = getelementptr [2 x uint]* %buffer, int 0, int 1 ; <uint*> [#uses=1]
store uint %tmp.3, uint* %tmp.5
%tmp.9 = cast [2 x uint]* %buffer to double* ; <double*> [#uses=1]
%tmp.10 = load double* %tmp.9 ; <double> [#uses=1]
%tmp.13 = load double* cast (long* %signed_bias to double*) ; <double> [#uses=1]
%tmp.14 = sub double %tmp.10, %tmp.13 ; <double> [#uses=1]
ret double %tmp.14
}
internal fastcc double %_Z2g2j(uint %x) {
entry:
%buffer = alloca [2 x uint] ; <[2 x uint]*> [#uses=3]
%tmp.0 = getelementptr [2 x uint]* %buffer, int 0, int 0 ; <uint*> [#uses=1]
store uint 1127219200, uint* %tmp.0
%tmp.1 = getelementptr [2 x uint]* %buffer, int 0, int 1 ; <uint*> [#uses=1]
store uint %x, uint* %tmp.1
%tmp.4 = cast [2 x uint]* %buffer to double* ; <double*> [#uses=1]
%tmp.5 = load double* %tmp.4 ; <double> [#uses=1]
%tmp.8 = load double* cast (long* %unsigned_bias to double*) ; <double> [#uses=1]
%tmp.9 = sub double %tmp.5, %tmp.8 ; <double> [#uses=1]
ret double %tmp.9
}
internal fastcc float %_Z2g3i(int %x) {
entry:
%buffer = alloca [2 x uint] ; <[2 x uint]*> [#uses=3]
%tmp.0 = getelementptr [2 x uint]* %buffer, int 0, int 0 ; <uint*> [#uses=1]
store uint 1127219200, uint* %tmp.0
%tmp.2 = cast int %x to uint ; <uint> [#uses=1]
%tmp.3 = xor uint %tmp.2, 2147483648 ; <uint> [#uses=1]
%tmp.5 = getelementptr [2 x uint]* %buffer, int 0, int 1 ; <uint*> [#uses=1]
store uint %tmp.3, uint* %tmp.5
%tmp.9 = cast [2 x uint]* %buffer to double* ; <double*> [#uses=1]
%tmp.10 = load double* %tmp.9 ; <double> [#uses=1]
%tmp.13 = load double* cast (long* %signed_bias to double*) ; <double> [#uses=1]
%tmp.14 = sub double %tmp.10, %tmp.13 ; <double> [#uses=1]
%tmp.16 = cast double %tmp.14 to float ; <float> [#uses=1]
ret float %tmp.16
}
internal fastcc float %_Z2g4j(uint %x) {
entry:
%buffer = alloca [2 x uint] ; <[2 x uint]*> [#uses=3]
%tmp.0 = getelementptr [2 x uint]* %buffer, int 0, int 0 ; <uint*> [#uses=1]
store uint 1127219200, uint* %tmp.0
%tmp.1 = getelementptr [2 x uint]* %buffer, int 0, int 1 ; <uint*> [#uses=1]
store uint %x, uint* %tmp.1
%tmp.4 = cast [2 x uint]* %buffer to double* ; <double*> [#uses=1]
%tmp.5 = load double* %tmp.4 ; <double> [#uses=1]
%tmp.8 = load double* cast (long* %unsigned_bias to double*) ; <double> [#uses=1]
%tmp.9 = sub double %tmp.5, %tmp.8 ; <double> [#uses=1]
%tmp.11 = cast double %tmp.9 to float ; <float> [#uses=1]
ret float %tmp.11
}
PowerPC Code:
.machine ppc970
.const
.align 2
.CPIl1__Z2f1i_0: ; float 0x4330000080000000
.long 1501560836 ; float 4.5036e+15
.text
.align 2
.globl l1__Z2f1i
l1__Z2f1i:
.LBBl1__Z2f1i_0: ; entry
xoris r2, r3, 32768
stw r2, -4(r1)
lis r2, 17200
stw r2, -8(r1)
lfd f0, -8(r1)
lis r2, ha16(.CPIl1__Z2f1i_0)
lfs f1, lo16(.CPIl1__Z2f1i_0)(r2)
fsub f1, f0, f1
blr
.const
.align 2
.CPIl2__Z2f2j_0: ; float 0x4330000000000000
.long 1501560832 ; float 4.5036e+15
.text
.align 2
.globl l2__Z2f2j
l2__Z2f2j:
.LBBl2__Z2f2j_0: ; entry
stw r3, -4(r1)
lis r2, 17200
stw r2, -8(r1)
lfd f0, -8(r1)
lis r2, ha16(.CPIl2__Z2f2j_0)
lfs f1, lo16(.CPIl2__Z2f2j_0)(r2)
fsub f1, f0, f1
blr
.const
.align 2
.CPIl3__Z2f3i_0: ; float 0x4330000080000000
.long 1501560836 ; float 4.5036e+15
.text
.align 2
.globl l3__Z2f3i
l3__Z2f3i:
.LBBl3__Z2f3i_0: ; entry
xoris r2, r3, 32768
stw r2, -4(r1)
lis r2, 17200
stw r2, -8(r1)
lfd f0, -8(r1)
lis r2, ha16(.CPIl3__Z2f3i_0)
lfs f1, lo16(.CPIl3__Z2f3i_0)(r2)
fsub f0, f0, f1
frsp f1, f0
blr
.const
.align 2
.CPIl4__Z2f4j_0: ; float 0x4330000000000000
.long 1501560832 ; float 4.5036e+15
.text
.align 2
.globl l4__Z2f4j
l4__Z2f4j:
.LBBl4__Z2f4j_0: ; entry
stw r3, -4(r1)
lis r2, 17200
stw r2, -8(r1)
lfd f0, -8(r1)
lis r2, ha16(.CPIl4__Z2f4j_0)
lfs f1, lo16(.CPIl4__Z2f4j_0)(r2)
fsub f0, f0, f1
frsp f1, f0
blr
llvm-svn: 22814
2005-08-17 08:39:29 +08:00
|
|
|
// final result
|
|
|
|
SDOperand Result;
|
|
|
|
// handle final rounding
|
|
|
|
if (DestVT == MVT::f64) {
|
|
|
|
// do nothing
|
|
|
|
Result = Sub;
|
|
|
|
} else {
|
|
|
|
// if f32 then cast to f32
|
|
|
|
Result = DAG.getNode(ISD::FP_ROUND, MVT::f32, Sub);
|
|
|
|
}
|
2005-12-20 08:53:54 +08:00
|
|
|
return LegalizeOp(Result);
|
Added generic code expansion for [signed|unsigned] i32 to [f32|f64] casts in the
legalizer. PowerPC now uses this expansion instead of ISel version.
Example:
// signed integer to double conversion
double f1(signed x) {
return (double)x;
}
// unsigned integer to double conversion
double f2(unsigned x) {
return (double)x;
}
// signed integer to float conversion
float f3(signed x) {
return (float)x;
}
// unsigned integer to float conversion
float f4(unsigned x) {
return (float)x;
}
Byte Code:
internal fastcc double %_Z2f1i(int %x) {
entry:
%tmp.1 = cast int %x to double ; <double> [#uses=1]
ret double %tmp.1
}
internal fastcc double %_Z2f2j(uint %x) {
entry:
%tmp.1 = cast uint %x to double ; <double> [#uses=1]
ret double %tmp.1
}
internal fastcc float %_Z2f3i(int %x) {
entry:
%tmp.1 = cast int %x to float ; <float> [#uses=1]
ret float %tmp.1
}
internal fastcc float %_Z2f4j(uint %x) {
entry:
%tmp.1 = cast uint %x to float ; <float> [#uses=1]
ret float %tmp.1
}
internal fastcc double %_Z2g1i(int %x) {
entry:
%buffer = alloca [2 x uint] ; <[2 x uint]*> [#uses=3]
%tmp.0 = getelementptr [2 x uint]* %buffer, int 0, int 0 ; <uint*> [#uses=1]
store uint 1127219200, uint* %tmp.0
%tmp.2 = cast int %x to uint ; <uint> [#uses=1]
%tmp.3 = xor uint %tmp.2, 2147483648 ; <uint> [#uses=1]
%tmp.5 = getelementptr [2 x uint]* %buffer, int 0, int 1 ; <uint*> [#uses=1]
store uint %tmp.3, uint* %tmp.5
%tmp.9 = cast [2 x uint]* %buffer to double* ; <double*> [#uses=1]
%tmp.10 = load double* %tmp.9 ; <double> [#uses=1]
%tmp.13 = load double* cast (long* %signed_bias to double*) ; <double> [#uses=1]
%tmp.14 = sub double %tmp.10, %tmp.13 ; <double> [#uses=1]
ret double %tmp.14
}
internal fastcc double %_Z2g2j(uint %x) {
entry:
%buffer = alloca [2 x uint] ; <[2 x uint]*> [#uses=3]
%tmp.0 = getelementptr [2 x uint]* %buffer, int 0, int 0 ; <uint*> [#uses=1]
store uint 1127219200, uint* %tmp.0
%tmp.1 = getelementptr [2 x uint]* %buffer, int 0, int 1 ; <uint*> [#uses=1]
store uint %x, uint* %tmp.1
%tmp.4 = cast [2 x uint]* %buffer to double* ; <double*> [#uses=1]
%tmp.5 = load double* %tmp.4 ; <double> [#uses=1]
%tmp.8 = load double* cast (long* %unsigned_bias to double*) ; <double> [#uses=1]
%tmp.9 = sub double %tmp.5, %tmp.8 ; <double> [#uses=1]
ret double %tmp.9
}
internal fastcc float %_Z2g3i(int %x) {
entry:
%buffer = alloca [2 x uint] ; <[2 x uint]*> [#uses=3]
%tmp.0 = getelementptr [2 x uint]* %buffer, int 0, int 0 ; <uint*> [#uses=1]
store uint 1127219200, uint* %tmp.0
%tmp.2 = cast int %x to uint ; <uint> [#uses=1]
%tmp.3 = xor uint %tmp.2, 2147483648 ; <uint> [#uses=1]
%tmp.5 = getelementptr [2 x uint]* %buffer, int 0, int 1 ; <uint*> [#uses=1]
store uint %tmp.3, uint* %tmp.5
%tmp.9 = cast [2 x uint]* %buffer to double* ; <double*> [#uses=1]
%tmp.10 = load double* %tmp.9 ; <double> [#uses=1]
%tmp.13 = load double* cast (long* %signed_bias to double*) ; <double> [#uses=1]
%tmp.14 = sub double %tmp.10, %tmp.13 ; <double> [#uses=1]
%tmp.16 = cast double %tmp.14 to float ; <float> [#uses=1]
ret float %tmp.16
}
internal fastcc float %_Z2g4j(uint %x) {
entry:
%buffer = alloca [2 x uint] ; <[2 x uint]*> [#uses=3]
%tmp.0 = getelementptr [2 x uint]* %buffer, int 0, int 0 ; <uint*> [#uses=1]
store uint 1127219200, uint* %tmp.0
%tmp.1 = getelementptr [2 x uint]* %buffer, int 0, int 1 ; <uint*> [#uses=1]
store uint %x, uint* %tmp.1
%tmp.4 = cast [2 x uint]* %buffer to double* ; <double*> [#uses=1]
%tmp.5 = load double* %tmp.4 ; <double> [#uses=1]
%tmp.8 = load double* cast (long* %unsigned_bias to double*) ; <double> [#uses=1]
%tmp.9 = sub double %tmp.5, %tmp.8 ; <double> [#uses=1]
%tmp.11 = cast double %tmp.9 to float ; <float> [#uses=1]
ret float %tmp.11
}
PowerPC Code:
.machine ppc970
.const
.align 2
.CPIl1__Z2f1i_0: ; float 0x4330000080000000
.long 1501560836 ; float 4.5036e+15
.text
.align 2
.globl l1__Z2f1i
l1__Z2f1i:
.LBBl1__Z2f1i_0: ; entry
xoris r2, r3, 32768
stw r2, -4(r1)
lis r2, 17200
stw r2, -8(r1)
lfd f0, -8(r1)
lis r2, ha16(.CPIl1__Z2f1i_0)
lfs f1, lo16(.CPIl1__Z2f1i_0)(r2)
fsub f1, f0, f1
blr
.const
.align 2
.CPIl2__Z2f2j_0: ; float 0x4330000000000000
.long 1501560832 ; float 4.5036e+15
.text
.align 2
.globl l2__Z2f2j
l2__Z2f2j:
.LBBl2__Z2f2j_0: ; entry
stw r3, -4(r1)
lis r2, 17200
stw r2, -8(r1)
lfd f0, -8(r1)
lis r2, ha16(.CPIl2__Z2f2j_0)
lfs f1, lo16(.CPIl2__Z2f2j_0)(r2)
fsub f1, f0, f1
blr
.const
.align 2
.CPIl3__Z2f3i_0: ; float 0x4330000080000000
.long 1501560836 ; float 4.5036e+15
.text
.align 2
.globl l3__Z2f3i
l3__Z2f3i:
.LBBl3__Z2f3i_0: ; entry
xoris r2, r3, 32768
stw r2, -4(r1)
lis r2, 17200
stw r2, -8(r1)
lfd f0, -8(r1)
lis r2, ha16(.CPIl3__Z2f3i_0)
lfs f1, lo16(.CPIl3__Z2f3i_0)(r2)
fsub f0, f0, f1
frsp f1, f0
blr
.const
.align 2
.CPIl4__Z2f4j_0: ; float 0x4330000000000000
.long 1501560832 ; float 4.5036e+15
.text
.align 2
.globl l4__Z2f4j
l4__Z2f4j:
.LBBl4__Z2f4j_0: ; entry
stw r3, -4(r1)
lis r2, 17200
stw r2, -8(r1)
lfd f0, -8(r1)
lis r2, ha16(.CPIl4__Z2f4j_0)
lfs f1, lo16(.CPIl4__Z2f4j_0)(r2)
fsub f0, f0, f1
frsp f1, f0
blr
llvm-svn: 22814
2005-08-17 08:39:29 +08:00
|
|
|
}
|
|
|
|
assert(!isSigned && "Legalize cannot Expand SINT_TO_FP for i64 yet");
|
2005-07-16 08:19:57 +08:00
|
|
|
SDOperand Tmp1 = DAG.getNode(ISD::SINT_TO_FP, DestVT, Op0);
|
2005-07-27 14:12:32 +08:00
|
|
|
|
2005-08-10 04:20:18 +08:00
|
|
|
SDOperand SignSet = DAG.getSetCC(TLI.getSetCCResultTy(), Op0,
|
|
|
|
DAG.getConstant(0, Op0.getValueType()),
|
|
|
|
ISD::SETLT);
|
2005-07-16 08:19:57 +08:00
|
|
|
SDOperand Zero = getIntPtrConstant(0), Four = getIntPtrConstant(4);
|
|
|
|
SDOperand CstOffset = DAG.getNode(ISD::SELECT, Zero.getValueType(),
|
|
|
|
SignSet, Four, Zero);
|
2005-07-27 14:12:32 +08:00
|
|
|
|
Added generic code expansion for [signed|unsigned] i32 to [f32|f64] casts in the
legalizer. PowerPC now uses this expansion instead of ISel version.
Example:
// signed integer to double conversion
double f1(signed x) {
return (double)x;
}
// unsigned integer to double conversion
double f2(unsigned x) {
return (double)x;
}
// signed integer to float conversion
float f3(signed x) {
return (float)x;
}
// unsigned integer to float conversion
float f4(unsigned x) {
return (float)x;
}
Byte Code:
internal fastcc double %_Z2f1i(int %x) {
entry:
%tmp.1 = cast int %x to double ; <double> [#uses=1]
ret double %tmp.1
}
internal fastcc double %_Z2f2j(uint %x) {
entry:
%tmp.1 = cast uint %x to double ; <double> [#uses=1]
ret double %tmp.1
}
internal fastcc float %_Z2f3i(int %x) {
entry:
%tmp.1 = cast int %x to float ; <float> [#uses=1]
ret float %tmp.1
}
internal fastcc float %_Z2f4j(uint %x) {
entry:
%tmp.1 = cast uint %x to float ; <float> [#uses=1]
ret float %tmp.1
}
internal fastcc double %_Z2g1i(int %x) {
entry:
%buffer = alloca [2 x uint] ; <[2 x uint]*> [#uses=3]
%tmp.0 = getelementptr [2 x uint]* %buffer, int 0, int 0 ; <uint*> [#uses=1]
store uint 1127219200, uint* %tmp.0
%tmp.2 = cast int %x to uint ; <uint> [#uses=1]
%tmp.3 = xor uint %tmp.2, 2147483648 ; <uint> [#uses=1]
%tmp.5 = getelementptr [2 x uint]* %buffer, int 0, int 1 ; <uint*> [#uses=1]
store uint %tmp.3, uint* %tmp.5
%tmp.9 = cast [2 x uint]* %buffer to double* ; <double*> [#uses=1]
%tmp.10 = load double* %tmp.9 ; <double> [#uses=1]
%tmp.13 = load double* cast (long* %signed_bias to double*) ; <double> [#uses=1]
%tmp.14 = sub double %tmp.10, %tmp.13 ; <double> [#uses=1]
ret double %tmp.14
}
internal fastcc double %_Z2g2j(uint %x) {
entry:
%buffer = alloca [2 x uint] ; <[2 x uint]*> [#uses=3]
%tmp.0 = getelementptr [2 x uint]* %buffer, int 0, int 0 ; <uint*> [#uses=1]
store uint 1127219200, uint* %tmp.0
%tmp.1 = getelementptr [2 x uint]* %buffer, int 0, int 1 ; <uint*> [#uses=1]
store uint %x, uint* %tmp.1
%tmp.4 = cast [2 x uint]* %buffer to double* ; <double*> [#uses=1]
%tmp.5 = load double* %tmp.4 ; <double> [#uses=1]
%tmp.8 = load double* cast (long* %unsigned_bias to double*) ; <double> [#uses=1]
%tmp.9 = sub double %tmp.5, %tmp.8 ; <double> [#uses=1]
ret double %tmp.9
}
internal fastcc float %_Z2g3i(int %x) {
entry:
%buffer = alloca [2 x uint] ; <[2 x uint]*> [#uses=3]
%tmp.0 = getelementptr [2 x uint]* %buffer, int 0, int 0 ; <uint*> [#uses=1]
store uint 1127219200, uint* %tmp.0
%tmp.2 = cast int %x to uint ; <uint> [#uses=1]
%tmp.3 = xor uint %tmp.2, 2147483648 ; <uint> [#uses=1]
%tmp.5 = getelementptr [2 x uint]* %buffer, int 0, int 1 ; <uint*> [#uses=1]
store uint %tmp.3, uint* %tmp.5
%tmp.9 = cast [2 x uint]* %buffer to double* ; <double*> [#uses=1]
%tmp.10 = load double* %tmp.9 ; <double> [#uses=1]
%tmp.13 = load double* cast (long* %signed_bias to double*) ; <double> [#uses=1]
%tmp.14 = sub double %tmp.10, %tmp.13 ; <double> [#uses=1]
%tmp.16 = cast double %tmp.14 to float ; <float> [#uses=1]
ret float %tmp.16
}
internal fastcc float %_Z2g4j(uint %x) {
entry:
%buffer = alloca [2 x uint] ; <[2 x uint]*> [#uses=3]
%tmp.0 = getelementptr [2 x uint]* %buffer, int 0, int 0 ; <uint*> [#uses=1]
store uint 1127219200, uint* %tmp.0
%tmp.1 = getelementptr [2 x uint]* %buffer, int 0, int 1 ; <uint*> [#uses=1]
store uint %x, uint* %tmp.1
%tmp.4 = cast [2 x uint]* %buffer to double* ; <double*> [#uses=1]
%tmp.5 = load double* %tmp.4 ; <double> [#uses=1]
%tmp.8 = load double* cast (long* %unsigned_bias to double*) ; <double> [#uses=1]
%tmp.9 = sub double %tmp.5, %tmp.8 ; <double> [#uses=1]
%tmp.11 = cast double %tmp.9 to float ; <float> [#uses=1]
ret float %tmp.11
}
PowerPC Code:
.machine ppc970
.const
.align 2
.CPIl1__Z2f1i_0: ; float 0x4330000080000000
.long 1501560836 ; float 4.5036e+15
.text
.align 2
.globl l1__Z2f1i
l1__Z2f1i:
.LBBl1__Z2f1i_0: ; entry
xoris r2, r3, 32768
stw r2, -4(r1)
lis r2, 17200
stw r2, -8(r1)
lfd f0, -8(r1)
lis r2, ha16(.CPIl1__Z2f1i_0)
lfs f1, lo16(.CPIl1__Z2f1i_0)(r2)
fsub f1, f0, f1
blr
.const
.align 2
.CPIl2__Z2f2j_0: ; float 0x4330000000000000
.long 1501560832 ; float 4.5036e+15
.text
.align 2
.globl l2__Z2f2j
l2__Z2f2j:
.LBBl2__Z2f2j_0: ; entry
stw r3, -4(r1)
lis r2, 17200
stw r2, -8(r1)
lfd f0, -8(r1)
lis r2, ha16(.CPIl2__Z2f2j_0)
lfs f1, lo16(.CPIl2__Z2f2j_0)(r2)
fsub f1, f0, f1
blr
.const
.align 2
.CPIl3__Z2f3i_0: ; float 0x4330000080000000
.long 1501560836 ; float 4.5036e+15
.text
.align 2
.globl l3__Z2f3i
l3__Z2f3i:
.LBBl3__Z2f3i_0: ; entry
xoris r2, r3, 32768
stw r2, -4(r1)
lis r2, 17200
stw r2, -8(r1)
lfd f0, -8(r1)
lis r2, ha16(.CPIl3__Z2f3i_0)
lfs f1, lo16(.CPIl3__Z2f3i_0)(r2)
fsub f0, f0, f1
frsp f1, f0
blr
.const
.align 2
.CPIl4__Z2f4j_0: ; float 0x4330000000000000
.long 1501560832 ; float 4.5036e+15
.text
.align 2
.globl l4__Z2f4j
l4__Z2f4j:
.LBBl4__Z2f4j_0: ; entry
stw r3, -4(r1)
lis r2, 17200
stw r2, -8(r1)
lfd f0, -8(r1)
lis r2, ha16(.CPIl4__Z2f4j_0)
lfs f1, lo16(.CPIl4__Z2f4j_0)(r2)
fsub f0, f0, f1
frsp f1, f0
blr
llvm-svn: 22814
2005-08-17 08:39:29 +08:00
|
|
|
// If the sign bit of the integer is set, the large number will be treated
|
|
|
|
// as a negative number. To counteract this, the dynamic code adds an
|
|
|
|
// offset depending on the data type.
|
2005-07-18 12:31:14 +08:00
|
|
|
uint64_t FF;
|
|
|
|
switch (Op0.getValueType()) {
|
|
|
|
default: assert(0 && "Unsupported integer type!");
|
|
|
|
case MVT::i8 : FF = 0x43800000ULL; break; // 2^8 (as a float)
|
|
|
|
case MVT::i16: FF = 0x47800000ULL; break; // 2^16 (as a float)
|
|
|
|
case MVT::i32: FF = 0x4F800000ULL; break; // 2^32 (as a float)
|
|
|
|
case MVT::i64: FF = 0x5F800000ULL; break; // 2^64 (as a float)
|
|
|
|
}
|
2005-07-16 08:19:57 +08:00
|
|
|
if (TLI.isLittleEndian()) FF <<= 32;
|
|
|
|
static Constant *FudgeFactor = ConstantUInt::get(Type::ULongTy, FF);
|
2005-07-27 14:12:32 +08:00
|
|
|
|
2005-08-27 01:15:30 +08:00
|
|
|
SDOperand CPIdx = DAG.getConstantPool(FudgeFactor, TLI.getPointerTy());
|
2005-07-16 08:19:57 +08:00
|
|
|
CPIdx = DAG.getNode(ISD::ADD, TLI.getPointerTy(), CPIdx, CstOffset);
|
|
|
|
SDOperand FudgeInReg;
|
|
|
|
if (DestVT == MVT::f32)
|
|
|
|
FudgeInReg = DAG.getLoad(MVT::f32, DAG.getEntryNode(), CPIdx,
|
|
|
|
DAG.getSrcValue(NULL));
|
|
|
|
else {
|
|
|
|
assert(DestVT == MVT::f64 && "Unexpected conversion");
|
|
|
|
FudgeInReg = LegalizeOp(DAG.getExtLoad(ISD::EXTLOAD, MVT::f64,
|
|
|
|
DAG.getEntryNode(), CPIdx,
|
|
|
|
DAG.getSrcValue(NULL), MVT::f32));
|
|
|
|
}
|
2005-07-27 14:12:32 +08:00
|
|
|
|
2005-12-20 08:53:54 +08:00
|
|
|
return LegalizeOp(DAG.getNode(ISD::FADD, DestVT, Tmp1, FudgeInReg));
|
2005-07-16 08:19:57 +08:00
|
|
|
}
|
|
|
|
|
2005-08-17 02:17:10 +08:00
|
|
|
/// PromoteLegalINT_TO_FP - This function is responsible for legalizing a
|
2005-07-29 08:11:56 +08:00
|
|
|
/// *INT_TO_FP operation of the specified operand when the target requests that
|
2005-07-16 08:19:57 +08:00
|
|
|
/// we promote it. At this point, we know that the result and operand types are
|
|
|
|
/// legal for the target, and that there is a legal UINT_TO_FP or SINT_TO_FP
|
|
|
|
/// operation that takes a larger input.
|
Teach the legalizer how to promote SINT_TO_FP to a wider SINT_TO_FP that
the target natively supports. This eliminates some special-case code from
the x86 backend and generates better code as well.
For an i8 to f64 conversion, before & after:
_x87 before:
subl $2, %esp
movb 6(%esp), %al
movsbw %al, %ax
movw %ax, (%esp)
filds (%esp)
addl $2, %esp
ret
_x87 after:
subl $2, %esp
movsbw 6(%esp), %ax
movw %ax, (%esp)
filds (%esp)
addl $2, %esp
ret
_sse before:
subl $12, %esp
movb 16(%esp), %al
movsbl %al, %eax
cvtsi2sd %eax, %xmm0
addl $12, %esp
ret
_sse after:
subl $12, %esp
movsbl 16(%esp), %eax
cvtsi2sd %eax, %xmm0
addl $12, %esp
ret
llvm-svn: 22452
2005-07-16 10:02:34 +08:00
|
|
|
SDOperand SelectionDAGLegalize::PromoteLegalINT_TO_FP(SDOperand LegalOp,
|
|
|
|
MVT::ValueType DestVT,
|
|
|
|
bool isSigned) {
|
2005-07-16 08:19:57 +08:00
|
|
|
// First step, figure out the appropriate *INT_TO_FP operation to use.
|
|
|
|
MVT::ValueType NewInTy = LegalOp.getValueType();
|
2005-07-27 14:12:32 +08:00
|
|
|
|
2005-07-16 08:19:57 +08:00
|
|
|
unsigned OpToUse = 0;
|
2005-07-27 14:12:32 +08:00
|
|
|
|
2005-07-16 08:19:57 +08:00
|
|
|
// Scan for the appropriate larger type to use.
|
|
|
|
while (1) {
|
|
|
|
NewInTy = (MVT::ValueType)(NewInTy+1);
|
|
|
|
assert(MVT::isInteger(NewInTy) && "Ran out of possibilities!");
|
2005-07-27 14:12:32 +08:00
|
|
|
|
2005-07-16 08:19:57 +08:00
|
|
|
// If the target supports SINT_TO_FP of this type, use it.
|
|
|
|
switch (TLI.getOperationAction(ISD::SINT_TO_FP, NewInTy)) {
|
|
|
|
default: break;
|
|
|
|
case TargetLowering::Legal:
|
2005-08-25 00:35:28 +08:00
|
|
|
if (!TLI.isTypeLegal(NewInTy))
|
2005-07-16 08:19:57 +08:00
|
|
|
break; // Can't use this datatype.
|
|
|
|
// FALL THROUGH.
|
|
|
|
case TargetLowering::Custom:
|
|
|
|
OpToUse = ISD::SINT_TO_FP;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
if (OpToUse) break;
|
Teach the legalizer how to promote SINT_TO_FP to a wider SINT_TO_FP that
the target natively supports. This eliminates some special-case code from
the x86 backend and generates better code as well.
For an i8 to f64 conversion, before & after:
_x87 before:
subl $2, %esp
movb 6(%esp), %al
movsbw %al, %ax
movw %ax, (%esp)
filds (%esp)
addl $2, %esp
ret
_x87 after:
subl $2, %esp
movsbw 6(%esp), %ax
movw %ax, (%esp)
filds (%esp)
addl $2, %esp
ret
_sse before:
subl $12, %esp
movb 16(%esp), %al
movsbl %al, %eax
cvtsi2sd %eax, %xmm0
addl $12, %esp
ret
_sse after:
subl $12, %esp
movsbl 16(%esp), %eax
cvtsi2sd %eax, %xmm0
addl $12, %esp
ret
llvm-svn: 22452
2005-07-16 10:02:34 +08:00
|
|
|
if (isSigned) continue;
|
2005-07-27 14:12:32 +08:00
|
|
|
|
2005-07-16 08:19:57 +08:00
|
|
|
// If the target supports UINT_TO_FP of this type, use it.
|
|
|
|
switch (TLI.getOperationAction(ISD::UINT_TO_FP, NewInTy)) {
|
|
|
|
default: break;
|
|
|
|
case TargetLowering::Legal:
|
2005-08-25 00:35:28 +08:00
|
|
|
if (!TLI.isTypeLegal(NewInTy))
|
2005-07-16 08:19:57 +08:00
|
|
|
break; // Can't use this datatype.
|
|
|
|
// FALL THROUGH.
|
|
|
|
case TargetLowering::Custom:
|
|
|
|
OpToUse = ISD::UINT_TO_FP;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
if (OpToUse) break;
|
2005-07-27 14:12:32 +08:00
|
|
|
|
2005-07-16 08:19:57 +08:00
|
|
|
// Otherwise, try a larger type.
|
|
|
|
}
|
|
|
|
|
|
|
|
// Okay, we found the operation and type to use. Zero extend our input to the
|
|
|
|
// desired type then run the operation on it.
|
2005-12-20 08:53:54 +08:00
|
|
|
SDOperand N = DAG.getNode(OpToUse, DestVT,
|
Teach the legalizer how to promote SINT_TO_FP to a wider SINT_TO_FP that
the target natively supports. This eliminates some special-case code from
the x86 backend and generates better code as well.
For an i8 to f64 conversion, before & after:
_x87 before:
subl $2, %esp
movb 6(%esp), %al
movsbw %al, %ax
movw %ax, (%esp)
filds (%esp)
addl $2, %esp
ret
_x87 after:
subl $2, %esp
movsbw 6(%esp), %ax
movw %ax, (%esp)
filds (%esp)
addl $2, %esp
ret
_sse before:
subl $12, %esp
movb 16(%esp), %al
movsbl %al, %eax
cvtsi2sd %eax, %xmm0
addl $12, %esp
ret
_sse after:
subl $12, %esp
movsbl 16(%esp), %eax
cvtsi2sd %eax, %xmm0
addl $12, %esp
ret
llvm-svn: 22452
2005-07-16 10:02:34 +08:00
|
|
|
DAG.getNode(isSigned ? ISD::SIGN_EXTEND : ISD::ZERO_EXTEND,
|
|
|
|
NewInTy, LegalOp));
|
2005-12-20 08:53:54 +08:00
|
|
|
// Make sure to legalize any nodes we create here.
|
|
|
|
return LegalizeOp(N);
|
2005-07-16 08:19:57 +08:00
|
|
|
}
|
|
|
|
|
2005-07-29 08:11:56 +08:00
|
|
|
/// PromoteLegalFP_TO_INT - This function is responsible for legalizing a
|
|
|
|
/// FP_TO_*INT operation of the specified operand when the target requests that
|
|
|
|
/// we promote it. At this point, we know that the result and operand types are
|
|
|
|
/// legal for the target, and that there is a legal FP_TO_UINT or FP_TO_SINT
|
|
|
|
/// operation that returns a larger result.
|
|
|
|
SDOperand SelectionDAGLegalize::PromoteLegalFP_TO_INT(SDOperand LegalOp,
|
|
|
|
MVT::ValueType DestVT,
|
|
|
|
bool isSigned) {
|
|
|
|
// First step, figure out the appropriate FP_TO*INT operation to use.
|
|
|
|
MVT::ValueType NewOutTy = DestVT;
|
2005-07-31 02:33:25 +08:00
|
|
|
|
2005-07-29 08:11:56 +08:00
|
|
|
unsigned OpToUse = 0;
|
2005-07-31 02:33:25 +08:00
|
|
|
|
2005-07-29 08:11:56 +08:00
|
|
|
// Scan for the appropriate larger type to use.
|
|
|
|
while (1) {
|
|
|
|
NewOutTy = (MVT::ValueType)(NewOutTy+1);
|
|
|
|
assert(MVT::isInteger(NewOutTy) && "Ran out of possibilities!");
|
2005-07-31 02:33:25 +08:00
|
|
|
|
2005-07-29 08:11:56 +08:00
|
|
|
// If the target supports FP_TO_SINT returning this type, use it.
|
|
|
|
switch (TLI.getOperationAction(ISD::FP_TO_SINT, NewOutTy)) {
|
|
|
|
default: break;
|
|
|
|
case TargetLowering::Legal:
|
2005-08-25 00:35:28 +08:00
|
|
|
if (!TLI.isTypeLegal(NewOutTy))
|
2005-07-29 08:11:56 +08:00
|
|
|
break; // Can't use this datatype.
|
|
|
|
// FALL THROUGH.
|
|
|
|
case TargetLowering::Custom:
|
|
|
|
OpToUse = ISD::FP_TO_SINT;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
if (OpToUse) break;
|
2005-07-31 02:33:25 +08:00
|
|
|
|
2005-07-29 08:11:56 +08:00
|
|
|
// If the target supports FP_TO_UINT of this type, use it.
|
|
|
|
switch (TLI.getOperationAction(ISD::FP_TO_UINT, NewOutTy)) {
|
|
|
|
default: break;
|
|
|
|
case TargetLowering::Legal:
|
2005-08-25 00:35:28 +08:00
|
|
|
if (!TLI.isTypeLegal(NewOutTy))
|
2005-07-29 08:11:56 +08:00
|
|
|
break; // Can't use this datatype.
|
|
|
|
// FALL THROUGH.
|
|
|
|
case TargetLowering::Custom:
|
|
|
|
OpToUse = ISD::FP_TO_UINT;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
if (OpToUse) break;
|
2005-07-31 02:33:25 +08:00
|
|
|
|
2005-07-29 08:11:56 +08:00
|
|
|
// Otherwise, try a larger type.
|
|
|
|
}
|
2005-07-31 02:33:25 +08:00
|
|
|
|
2005-07-29 08:11:56 +08:00
|
|
|
// Okay, we found the operation and type to use. Truncate the result of the
|
|
|
|
// extended FP_TO_*INT operation to the desired size.
|
2005-12-20 08:53:54 +08:00
|
|
|
SDOperand N = DAG.getNode(ISD::TRUNCATE, DestVT,
|
|
|
|
DAG.getNode(OpToUse, NewOutTy, LegalOp));
|
|
|
|
// Make sure to legalize any nodes we create here in the next pass.
|
|
|
|
return LegalizeOp(N);
|
2005-07-29 08:11:56 +08:00
|
|
|
}
|
|
|
|
|
2005-10-06 09:20:27 +08:00
|
|
|
/// ComputeTopDownOrdering - Add the specified node to the Order list if it has
|
|
|
|
/// not been visited yet and if all of its operands have already been visited.
|
|
|
|
static void ComputeTopDownOrdering(SDNode *N, std::vector<SDNode*> &Order,
|
|
|
|
std::map<SDNode*, unsigned> &Visited) {
|
|
|
|
if (++Visited[N] != N->getNumOperands())
|
|
|
|
return; // Haven't visited all operands yet
|
|
|
|
|
|
|
|
Order.push_back(N);
|
|
|
|
|
|
|
|
if (N->hasOneUse()) { // Tail recurse in common case.
|
|
|
|
ComputeTopDownOrdering(*N->use_begin(), Order, Visited);
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
// Now that we have N in, add anything that uses it if all of their operands
|
|
|
|
// are now done.
|
|
|
|
for (SDNode::use_iterator UI = N->use_begin(), E = N->use_end(); UI != E;++UI)
|
|
|
|
ComputeTopDownOrdering(*UI, Order, Visited);
|
|
|
|
}
|
|
|
|
|
2005-07-29 08:11:56 +08:00
|
|
|
|
2005-01-07 15:47:09 +08:00
|
|
|
void SelectionDAGLegalize::LegalizeDAG() {
|
2005-10-03 01:49:46 +08:00
|
|
|
// The legalize process is inherently a bottom-up recursive process (users
|
|
|
|
// legalize their uses before themselves). Given infinite stack space, we
|
|
|
|
// could just start legalizing on the root and traverse the whole graph. In
|
|
|
|
// practice however, this causes us to run out of stack space on large basic
|
2005-10-06 09:20:27 +08:00
|
|
|
// blocks. To avoid this problem, compute an ordering of the nodes where each
|
|
|
|
// node is only legalized after all of its operands are legalized.
|
|
|
|
std::map<SDNode*, unsigned> Visited;
|
|
|
|
std::vector<SDNode*> Order;
|
2005-10-03 01:49:46 +08:00
|
|
|
|
2005-10-06 09:20:27 +08:00
|
|
|
// Compute ordering from all of the leaves in the graphs, those (like the
|
|
|
|
// entry node) that have no operands.
|
|
|
|
for (SelectionDAG::allnodes_iterator I = DAG.allnodes_begin(),
|
|
|
|
E = DAG.allnodes_end(); I != E; ++I) {
|
2005-11-10 07:47:37 +08:00
|
|
|
if (I->getNumOperands() == 0) {
|
|
|
|
Visited[I] = 0 - 1U;
|
|
|
|
ComputeTopDownOrdering(I, Order, Visited);
|
2005-10-03 01:49:46 +08:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2005-11-10 07:47:37 +08:00
|
|
|
assert(Order.size() == Visited.size() &&
|
|
|
|
Order.size() ==
|
|
|
|
(unsigned)std::distance(DAG.allnodes_begin(), DAG.allnodes_end()) &&
|
2005-10-06 09:20:27 +08:00
|
|
|
"Error: DAG is cyclic!");
|
|
|
|
Visited.clear();
|
2005-10-03 01:49:46 +08:00
|
|
|
|
2005-10-06 09:20:27 +08:00
|
|
|
for (unsigned i = 0, e = Order.size(); i != e; ++i) {
|
|
|
|
SDNode *N = Order[i];
|
|
|
|
switch (getTypeAction(N->getValueType(0))) {
|
|
|
|
default: assert(0 && "Bad type action!");
|
|
|
|
case Legal:
|
|
|
|
LegalizeOp(SDOperand(N, 0));
|
|
|
|
break;
|
|
|
|
case Promote:
|
|
|
|
PromoteOp(SDOperand(N, 0));
|
|
|
|
break;
|
|
|
|
case Expand: {
|
|
|
|
SDOperand X, Y;
|
|
|
|
ExpandOp(SDOperand(N, 0), X, Y);
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// Finally, it's possible the root changed. Get the new root.
|
2005-01-07 15:47:09 +08:00
|
|
|
SDOperand OldRoot = DAG.getRoot();
|
2005-10-06 09:20:27 +08:00
|
|
|
assert(LegalizedNodes.count(OldRoot) && "Root didn't get legalized?");
|
|
|
|
DAG.setRoot(LegalizedNodes[OldRoot]);
|
2005-01-07 15:47:09 +08:00
|
|
|
|
|
|
|
ExpandedNodes.clear();
|
|
|
|
LegalizedNodes.clear();
|
2005-01-16 09:11:45 +08:00
|
|
|
PromotedNodes.clear();
|
2005-01-07 15:47:09 +08:00
|
|
|
|
|
|
|
// Remove dead nodes now.
|
2005-01-08 05:09:37 +08:00
|
|
|
DAG.RemoveDeadNodes(OldRoot.Val);
|
2005-01-07 15:47:09 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
SDOperand SelectionDAGLegalize::LegalizeOp(SDOperand Op) {
|
2005-08-25 00:35:28 +08:00
|
|
|
assert(isTypeLegal(Op.getValueType()) &&
|
2005-01-09 04:35:13 +08:00
|
|
|
"Caller should expand or promote operands that are not legal!");
|
2005-05-13 00:53:42 +08:00
|
|
|
SDNode *Node = Op.Val;
|
2005-01-09 04:35:13 +08:00
|
|
|
|
2005-01-07 15:47:09 +08:00
|
|
|
// If this operation defines any values that cannot be represented in a
|
2005-01-09 04:35:13 +08:00
|
|
|
// register on this target, make sure to expand or promote them.
|
2005-05-13 00:53:42 +08:00
|
|
|
if (Node->getNumValues() > 1) {
|
|
|
|
for (unsigned i = 0, e = Node->getNumValues(); i != e; ++i)
|
|
|
|
switch (getTypeAction(Node->getValueType(i))) {
|
2005-01-07 15:47:09 +08:00
|
|
|
case Legal: break; // Nothing to do.
|
|
|
|
case Expand: {
|
|
|
|
SDOperand T1, T2;
|
|
|
|
ExpandOp(Op.getValue(i), T1, T2);
|
|
|
|
assert(LegalizedNodes.count(Op) &&
|
|
|
|
"Expansion didn't add legal operands!");
|
|
|
|
return LegalizedNodes[Op];
|
|
|
|
}
|
|
|
|
case Promote:
|
2005-01-15 13:21:40 +08:00
|
|
|
PromoteOp(Op.getValue(i));
|
|
|
|
assert(LegalizedNodes.count(Op) &&
|
|
|
|
"Expansion didn't add legal operands!");
|
|
|
|
return LegalizedNodes[Op];
|
2005-01-07 15:47:09 +08:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2005-05-13 00:53:42 +08:00
|
|
|
// Note that LegalizeOp may be reentered even from single-use nodes, which
|
|
|
|
// means that we always must cache transformed nodes.
|
2005-01-11 13:57:22 +08:00
|
|
|
std::map<SDOperand, SDOperand>::iterator I = LegalizedNodes.find(Op);
|
|
|
|
if (I != LegalizedNodes.end()) return I->second;
|
2005-01-07 15:47:09 +08:00
|
|
|
|
2005-08-11 04:51:12 +08:00
|
|
|
SDOperand Tmp1, Tmp2, Tmp3, Tmp4;
|
2005-01-07 15:47:09 +08:00
|
|
|
|
|
|
|
SDOperand Result = Op;
|
|
|
|
|
|
|
|
switch (Node->getOpcode()) {
|
|
|
|
default:
|
2005-05-14 14:34:48 +08:00
|
|
|
if (Node->getOpcode() >= ISD::BUILTIN_OP_END) {
|
|
|
|
// If this is a target node, legalize it by legalizing the operands then
|
|
|
|
// passing it through.
|
|
|
|
std::vector<SDOperand> Ops;
|
|
|
|
bool Changed = false;
|
|
|
|
for (unsigned i = 0, e = Node->getNumOperands(); i != e; ++i) {
|
|
|
|
Ops.push_back(LegalizeOp(Node->getOperand(i)));
|
|
|
|
Changed = Changed || Node->getOperand(i) != Ops.back();
|
|
|
|
}
|
|
|
|
if (Changed)
|
|
|
|
if (Node->getNumValues() == 1)
|
|
|
|
Result = DAG.getNode(Node->getOpcode(), Node->getValueType(0), Ops);
|
|
|
|
else {
|
|
|
|
std::vector<MVT::ValueType> VTs(Node->value_begin(),
|
|
|
|
Node->value_end());
|
|
|
|
Result = DAG.getNode(Node->getOpcode(), VTs, Ops);
|
|
|
|
}
|
|
|
|
|
|
|
|
for (unsigned i = 0, e = Node->getNumValues(); i != e; ++i)
|
|
|
|
AddLegalizedOperand(Op.getValue(i), Result.getValue(i));
|
|
|
|
return Result.getValue(Op.ResNo);
|
|
|
|
}
|
|
|
|
// Otherwise this is an unhandled builtin node. splat.
|
2005-01-07 15:47:09 +08:00
|
|
|
std::cerr << "NODE: "; Node->dump(); std::cerr << "\n";
|
|
|
|
assert(0 && "Do not know how to legalize this operator!");
|
|
|
|
abort();
|
|
|
|
case ISD::EntryToken:
|
|
|
|
case ISD::FrameIndex:
|
2005-10-06 09:20:27 +08:00
|
|
|
case ISD::TargetFrameIndex:
|
|
|
|
case ISD::Register:
|
|
|
|
case ISD::TargetConstant:
|
2005-12-10 10:36:00 +08:00
|
|
|
case ISD::TargetConstantPool:
|
2005-01-07 15:47:09 +08:00
|
|
|
case ISD::GlobalAddress:
|
2005-11-17 13:52:24 +08:00
|
|
|
case ISD::TargetGlobalAddress:
|
2005-01-08 05:45:56 +08:00
|
|
|
case ISD::ExternalSymbol:
|
2005-12-25 07:42:32 +08:00
|
|
|
case ISD::TargetExternalSymbol:
|
2005-01-15 06:38:01 +08:00
|
|
|
case ISD::ConstantPool: // Nothing to do.
|
2005-10-06 09:20:27 +08:00
|
|
|
case ISD::BasicBlock:
|
|
|
|
case ISD::CONDCODE:
|
|
|
|
case ISD::VALUETYPE:
|
|
|
|
case ISD::SRCVALUE:
|
2005-11-29 14:21:05 +08:00
|
|
|
case ISD::STRING:
|
2005-11-17 14:41:44 +08:00
|
|
|
switch (TLI.getOperationAction(Node->getOpcode(), Node->getValueType(0))) {
|
|
|
|
default: assert(0 && "This action is not supported yet!");
|
|
|
|
case TargetLowering::Custom: {
|
|
|
|
SDOperand Tmp = TLI.LowerOperation(Op, DAG);
|
|
|
|
if (Tmp.Val) {
|
|
|
|
Result = LegalizeOp(Tmp);
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
} // FALLTHROUGH if the target doesn't want to lower this op after all.
|
|
|
|
case TargetLowering::Legal:
|
|
|
|
assert(isTypeLegal(Node->getValueType(0)) && "This must be legal!");
|
|
|
|
break;
|
|
|
|
}
|
2005-01-07 15:47:09 +08:00
|
|
|
break;
|
2005-09-02 09:15:01 +08:00
|
|
|
case ISD::AssertSext:
|
|
|
|
case ISD::AssertZext:
|
|
|
|
Tmp1 = LegalizeOp(Node->getOperand(0));
|
|
|
|
if (Tmp1 != Node->getOperand(0))
|
|
|
|
Result = DAG.getNode(Node->getOpcode(), Node->getValueType(0), Tmp1,
|
|
|
|
Node->getOperand(1));
|
|
|
|
break;
|
2005-11-21 06:56:56 +08:00
|
|
|
case ISD::MERGE_VALUES:
|
|
|
|
return LegalizeOp(Node->getOperand(Op.ResNo));
|
2005-01-15 06:38:01 +08:00
|
|
|
case ISD::CopyFromReg:
|
|
|
|
Tmp1 = LegalizeOp(Node->getOperand(0));
|
2005-12-18 23:27:43 +08:00
|
|
|
Result = Op.getValue(0);
|
2005-12-18 23:36:21 +08:00
|
|
|
if (Node->getNumValues() == 2) {
|
2005-12-18 23:27:43 +08:00
|
|
|
if (Tmp1 != Node->getOperand(0))
|
|
|
|
Result = DAG.getCopyFromReg(Tmp1,
|
2005-08-17 05:55:35 +08:00
|
|
|
cast<RegisterSDNode>(Node->getOperand(1))->getReg(),
|
2005-12-18 23:27:43 +08:00
|
|
|
Node->getValueType(0));
|
|
|
|
} else {
|
2005-12-18 23:36:21 +08:00
|
|
|
assert(Node->getNumValues() == 3 && "Invalid copyfromreg!");
|
|
|
|
if (Node->getNumOperands() == 3)
|
|
|
|
Tmp2 = LegalizeOp(Node->getOperand(2));
|
|
|
|
if (Tmp1 != Node->getOperand(0) ||
|
|
|
|
(Node->getNumOperands() == 3 && Tmp2 != Node->getOperand(2)))
|
2005-12-18 23:27:43 +08:00
|
|
|
Result = DAG.getCopyFromReg(Tmp1,
|
|
|
|
cast<RegisterSDNode>(Node->getOperand(1))->getReg(),
|
|
|
|
Node->getValueType(0), Tmp2);
|
|
|
|
AddLegalizedOperand(Op.getValue(2), Result.getValue(2));
|
|
|
|
}
|
2005-01-28 14:27:38 +08:00
|
|
|
// Since CopyFromReg produces two values, make sure to remember that we
|
|
|
|
// legalized both of them.
|
|
|
|
AddLegalizedOperand(Op.getValue(0), Result);
|
|
|
|
AddLegalizedOperand(Op.getValue(1), Result.getValue(1));
|
|
|
|
return Result.getValue(Op.ResNo);
|
2005-04-02 06:34:39 +08:00
|
|
|
case ISD::UNDEF: {
|
|
|
|
MVT::ValueType VT = Op.getValueType();
|
|
|
|
switch (TLI.getOperationAction(ISD::UNDEF, VT)) {
|
2005-04-02 08:41:14 +08:00
|
|
|
default: assert(0 && "This action is not supported yet!");
|
|
|
|
case TargetLowering::Expand:
|
|
|
|
case TargetLowering::Promote:
|
2005-04-02 06:34:39 +08:00
|
|
|
if (MVT::isInteger(VT))
|
|
|
|
Result = DAG.getConstant(0, VT);
|
|
|
|
else if (MVT::isFloatingPoint(VT))
|
|
|
|
Result = DAG.getConstantFP(0, VT);
|
|
|
|
else
|
|
|
|
assert(0 && "Unknown value type!");
|
|
|
|
break;
|
2005-04-02 08:41:14 +08:00
|
|
|
case TargetLowering::Legal:
|
2005-04-02 06:34:39 +08:00
|
|
|
break;
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
}
|
2005-11-29 14:21:05 +08:00
|
|
|
|
|
|
|
case ISD::LOCATION:
|
|
|
|
assert(Node->getNumOperands() == 5 && "Invalid LOCATION node!");
|
|
|
|
Tmp1 = LegalizeOp(Node->getOperand(0)); // Legalize the input chain.
|
|
|
|
|
|
|
|
switch (TLI.getOperationAction(ISD::LOCATION, MVT::Other)) {
|
|
|
|
case TargetLowering::Promote:
|
|
|
|
default: assert(0 && "This action is not supported yet!");
|
2005-12-17 06:45:29 +08:00
|
|
|
case TargetLowering::Expand: {
|
2006-01-05 06:28:25 +08:00
|
|
|
MachineDebugInfo *DebugInfo = DAG.getMachineDebugInfo();
|
2006-01-05 09:25:28 +08:00
|
|
|
bool useDEBUG_LOC = TLI.isOperationLegal(ISD::DEBUG_LOC, MVT::Other);
|
|
|
|
bool useDEBUG_LABEL = TLI.isOperationLegal(ISD::DEBUG_LABEL, MVT::Other);
|
|
|
|
|
|
|
|
if (DebugInfo && (useDEBUG_LOC || useDEBUG_LABEL)) {
|
|
|
|
const std::string &FName =
|
2005-12-22 04:51:37 +08:00
|
|
|
cast<StringSDNode>(Node->getOperand(3))->getValue();
|
2006-01-05 09:25:28 +08:00
|
|
|
const std::string &DirName =
|
2005-12-22 04:51:37 +08:00
|
|
|
cast<StringSDNode>(Node->getOperand(4))->getValue();
|
2006-01-18 01:31:53 +08:00
|
|
|
unsigned SrcFile = DebugInfo->RecordSource(DirName, FName);
|
2006-01-05 09:25:28 +08:00
|
|
|
|
|
|
|
std::vector<SDOperand> Ops;
|
|
|
|
Ops.push_back(Tmp1); // chain
|
|
|
|
SDOperand LineOp = Node->getOperand(1);
|
|
|
|
SDOperand ColOp = Node->getOperand(2);
|
|
|
|
|
|
|
|
if (useDEBUG_LOC) {
|
|
|
|
Ops.push_back(LineOp); // line #
|
|
|
|
Ops.push_back(ColOp); // col #
|
|
|
|
Ops.push_back(DAG.getConstant(SrcFile, MVT::i32)); // source file id
|
|
|
|
Result = DAG.getNode(ISD::DEBUG_LOC, MVT::Other, Ops);
|
|
|
|
} else {
|
|
|
|
unsigned Line = dyn_cast<ConstantSDNode>(LineOp)->getValue();
|
|
|
|
unsigned Col = dyn_cast<ConstantSDNode>(ColOp)->getValue();
|
|
|
|
unsigned ID = DebugInfo->RecordLabel(Line, Col, SrcFile);
|
|
|
|
Ops.push_back(DAG.getConstant(ID, MVT::i32));
|
|
|
|
Result = DAG.getNode(ISD::DEBUG_LABEL, MVT::Other, Ops);
|
|
|
|
}
|
2005-12-22 04:51:37 +08:00
|
|
|
} else {
|
|
|
|
Result = Tmp1; // chain
|
|
|
|
}
|
2005-12-19 07:54:29 +08:00
|
|
|
Result = LegalizeOp(Result); // Relegalize new nodes.
|
2005-11-29 14:21:05 +08:00
|
|
|
break;
|
2005-12-19 07:54:29 +08:00
|
|
|
}
|
2005-11-29 14:21:05 +08:00
|
|
|
case TargetLowering::Legal:
|
2005-12-02 02:21:35 +08:00
|
|
|
if (Tmp1 != Node->getOperand(0) ||
|
|
|
|
getTypeAction(Node->getOperand(1).getValueType()) == Promote) {
|
2005-11-29 14:21:05 +08:00
|
|
|
std::vector<SDOperand> Ops;
|
|
|
|
Ops.push_back(Tmp1);
|
2005-12-02 02:21:35 +08:00
|
|
|
if (getTypeAction(Node->getOperand(1).getValueType()) == Legal) {
|
|
|
|
Ops.push_back(Node->getOperand(1)); // line # must be legal.
|
|
|
|
Ops.push_back(Node->getOperand(2)); // col # must be legal.
|
|
|
|
} else {
|
|
|
|
// Otherwise promote them.
|
|
|
|
Ops.push_back(PromoteOp(Node->getOperand(1)));
|
|
|
|
Ops.push_back(PromoteOp(Node->getOperand(2)));
|
|
|
|
}
|
2005-11-29 14:21:05 +08:00
|
|
|
Ops.push_back(Node->getOperand(3)); // filename must be legal.
|
|
|
|
Ops.push_back(Node->getOperand(4)); // working dir # must be legal.
|
|
|
|
Result = DAG.getNode(ISD::LOCATION, MVT::Other, Ops);
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
break;
|
2005-12-17 06:45:29 +08:00
|
|
|
|
|
|
|
case ISD::DEBUG_LOC:
|
2006-01-05 09:25:28 +08:00
|
|
|
assert(Node->getNumOperands() == 4 && "Invalid DEBUG_LOC node!");
|
2005-12-17 06:45:29 +08:00
|
|
|
switch (TLI.getOperationAction(ISD::DEBUG_LOC, MVT::Other)) {
|
|
|
|
case TargetLowering::Promote:
|
|
|
|
case TargetLowering::Expand:
|
|
|
|
default: assert(0 && "This action is not supported yet!");
|
2006-01-05 09:25:28 +08:00
|
|
|
case TargetLowering::Legal:
|
|
|
|
Tmp1 = LegalizeOp(Node->getOperand(0)); // Legalize the chain.
|
|
|
|
Tmp2 = LegalizeOp(Node->getOperand(1)); // Legalize the line #.
|
|
|
|
Tmp3 = LegalizeOp(Node->getOperand(2)); // Legalize the col #.
|
|
|
|
Tmp4 = LegalizeOp(Node->getOperand(3)); // Legalize the source file id.
|
|
|
|
|
|
|
|
if (Tmp1 != Node->getOperand(0) ||
|
|
|
|
Tmp2 != Node->getOperand(1) ||
|
|
|
|
Tmp3 != Node->getOperand(2) ||
|
|
|
|
Tmp4 != Node->getOperand(3)) {
|
|
|
|
Result = DAG.getNode(ISD::DEBUG_LOC,MVT::Other, Tmp1, Tmp2, Tmp3, Tmp4);
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
|
|
|
|
case ISD::DEBUG_LABEL:
|
|
|
|
assert(Node->getNumOperands() == 2 && "Invalid DEBUG_LABEL node!");
|
|
|
|
switch (TLI.getOperationAction(ISD::DEBUG_LABEL, MVT::Other)) {
|
|
|
|
case TargetLowering::Promote:
|
|
|
|
case TargetLowering::Expand:
|
|
|
|
default: assert(0 && "This action is not supported yet!");
|
|
|
|
case TargetLowering::Legal:
|
|
|
|
Tmp1 = LegalizeOp(Node->getOperand(0)); // Legalize the chain.
|
|
|
|
Tmp2 = LegalizeOp(Node->getOperand(1)); // Legalize the label id.
|
|
|
|
|
|
|
|
if (Tmp1 != Node->getOperand(0) ||
|
|
|
|
Tmp2 != Node->getOperand(1)) {
|
|
|
|
Result = DAG.getNode(ISD::DEBUG_LABEL, MVT::Other, Tmp1, Tmp2);
|
2005-12-17 06:45:29 +08:00
|
|
|
}
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
break;
|
2005-11-29 14:21:05 +08:00
|
|
|
|
2005-01-07 15:47:09 +08:00
|
|
|
case ISD::Constant:
|
|
|
|
// We know we don't need to expand constants here, constants only have one
|
|
|
|
// value and we check that it is fine above.
|
|
|
|
|
|
|
|
// FIXME: Maybe we should handle things like targets that don't support full
|
|
|
|
// 32-bit immediates?
|
|
|
|
break;
|
|
|
|
case ISD::ConstantFP: {
|
|
|
|
// Spill FP immediates to the constant pool if the target cannot directly
|
|
|
|
// codegen them. Targets often have some immediate values that can be
|
|
|
|
// efficiently generated into an FP register without a load. We explicitly
|
|
|
|
// leave these constants as ConstantFP nodes for the target to deal with.
|
|
|
|
|
|
|
|
ConstantFPSDNode *CFP = cast<ConstantFPSDNode>(Node);
|
|
|
|
|
|
|
|
// Check to see if this FP immediate is already legal.
|
|
|
|
bool isLegal = false;
|
|
|
|
for (TargetLowering::legal_fpimm_iterator I = TLI.legal_fpimm_begin(),
|
|
|
|
E = TLI.legal_fpimm_end(); I != E; ++I)
|
|
|
|
if (CFP->isExactlyValue(*I)) {
|
|
|
|
isLegal = true;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (!isLegal) {
|
|
|
|
// Otherwise we need to spill the constant to memory.
|
|
|
|
bool Extend = false;
|
|
|
|
|
|
|
|
// If a FP immediate is precise when represented as a float, we put it
|
|
|
|
// into the constant pool as a float, even if it's is statically typed
|
|
|
|
// as a double.
|
|
|
|
MVT::ValueType VT = CFP->getValueType(0);
|
|
|
|
bool isDouble = VT == MVT::f64;
|
|
|
|
ConstantFP *LLVMC = ConstantFP::get(isDouble ? Type::DoubleTy :
|
|
|
|
Type::FloatTy, CFP->getValue());
|
2005-01-29 06:58:25 +08:00
|
|
|
if (isDouble && CFP->isExactlyValue((float)CFP->getValue()) &&
|
|
|
|
// Only do this if the target has a native EXTLOAD instruction from
|
|
|
|
// f32.
|
2005-08-25 00:35:28 +08:00
|
|
|
TLI.isOperationLegal(ISD::EXTLOAD, MVT::f32)) {
|
2005-01-07 15:47:09 +08:00
|
|
|
LLVMC = cast<ConstantFP>(ConstantExpr::getCast(LLVMC, Type::FloatTy));
|
|
|
|
VT = MVT::f32;
|
|
|
|
Extend = true;
|
|
|
|
}
|
2005-04-22 06:36:52 +08:00
|
|
|
|
2005-12-10 10:36:00 +08:00
|
|
|
SDOperand CPIdx =
|
|
|
|
LegalizeOp(DAG.getConstantPool(LLVMC, TLI.getPointerTy()));
|
2005-01-16 13:06:12 +08:00
|
|
|
if (Extend) {
|
2005-07-10 09:55:33 +08:00
|
|
|
Result = DAG.getExtLoad(ISD::EXTLOAD, MVT::f64, DAG.getEntryNode(),
|
|
|
|
CPIdx, DAG.getSrcValue(NULL), MVT::f32);
|
2005-01-16 13:06:12 +08:00
|
|
|
} else {
|
2005-05-10 04:23:03 +08:00
|
|
|
Result = DAG.getLoad(VT, DAG.getEntryNode(), CPIdx,
|
|
|
|
DAG.getSrcValue(NULL));
|
2005-01-16 13:06:12 +08:00
|
|
|
}
|
2005-01-07 15:47:09 +08:00
|
|
|
}
|
|
|
|
break;
|
|
|
|
}
|
Teach the SelectionDAG ISel how to turn ConstantPacked values into
constant nodes with vector types. Also teach the asm printer how to print
ConstantPacked constant pool entries. This allows us to generate altivec
code such as the following, which adds a vector constantto a packed float.
LCPI1_0: <4 x float> < float 0.0e+0, float 0.0e+0, float 0.0e+0, float 1.0e+0 >
.space 4
.space 4
.space 4
.long 1065353216 ; float 1
.text
.align 4
.globl _foo
_foo:
lis r2, ha16(LCPI1_0)
la r2, lo16(LCPI1_0)(r2)
li r4, 0
lvx v0, r4, r2
lvx v1, r4, r3
vaddfp v0, v1, v0
stvx v0, r4, r3
blr
For the llvm code:
void %foo(<4 x float> * %a) {
entry:
%tmp1 = load <4 x float> * %a;
%tmp2 = add <4 x float> %tmp1, < float 0.0, float 0.0, float 0.0, float 1.0 >
store <4 x float> %tmp2, <4 x float> *%a
ret void
}
llvm-svn: 24616
2005-12-06 14:18:55 +08:00
|
|
|
case ISD::ConstantVec: {
|
|
|
|
// We assume that vector constants are not legal, and will be immediately
|
|
|
|
// spilled to the constant pool.
|
|
|
|
//
|
|
|
|
// FIXME: revisit this when we have some kind of mechanism by which targets
|
|
|
|
// can decided legality of vector constants, of which there may be very
|
|
|
|
// many.
|
|
|
|
//
|
|
|
|
// Create a ConstantPacked, and put it in the constant pool.
|
|
|
|
std::vector<Constant*> CV;
|
|
|
|
MVT::ValueType VT = Node->getValueType(0);
|
|
|
|
for (unsigned I = 0, E = Node->getNumOperands(); I < E; ++I) {
|
|
|
|
SDOperand OpN = Node->getOperand(I);
|
|
|
|
const Type* OpNTy = MVT::getTypeForValueType(OpN.getValueType());
|
|
|
|
if (MVT::isFloatingPoint(VT))
|
|
|
|
CV.push_back(ConstantFP::get(OpNTy,
|
|
|
|
cast<ConstantFPSDNode>(OpN)->getValue()));
|
|
|
|
else
|
|
|
|
CV.push_back(ConstantUInt::get(OpNTy,
|
|
|
|
cast<ConstantSDNode>(OpN)->getValue()));
|
|
|
|
}
|
|
|
|
Constant *CP = ConstantPacked::get(CV);
|
2005-12-13 11:03:23 +08:00
|
|
|
SDOperand CPIdx = LegalizeOp(DAG.getConstantPool(CP, TLI.getPointerTy()));
|
Teach the SelectionDAG ISel how to turn ConstantPacked values into
constant nodes with vector types. Also teach the asm printer how to print
ConstantPacked constant pool entries. This allows us to generate altivec
code such as the following, which adds a vector constantto a packed float.
LCPI1_0: <4 x float> < float 0.0e+0, float 0.0e+0, float 0.0e+0, float 1.0e+0 >
.space 4
.space 4
.space 4
.long 1065353216 ; float 1
.text
.align 4
.globl _foo
_foo:
lis r2, ha16(LCPI1_0)
la r2, lo16(LCPI1_0)(r2)
li r4, 0
lvx v0, r4, r2
lvx v1, r4, r3
vaddfp v0, v1, v0
stvx v0, r4, r3
blr
For the llvm code:
void %foo(<4 x float> * %a) {
entry:
%tmp1 = load <4 x float> * %a;
%tmp2 = add <4 x float> %tmp1, < float 0.0, float 0.0, float 0.0, float 1.0 >
store <4 x float> %tmp2, <4 x float> *%a
ret void
}
llvm-svn: 24616
2005-12-06 14:18:55 +08:00
|
|
|
Result = DAG.getLoad(VT, DAG.getEntryNode(), CPIdx, DAG.getSrcValue(NULL));
|
|
|
|
break;
|
|
|
|
}
|
2005-11-10 02:48:57 +08:00
|
|
|
case ISD::TokenFactor:
|
|
|
|
if (Node->getNumOperands() == 2) {
|
|
|
|
bool Changed = false;
|
|
|
|
SDOperand Op0 = LegalizeOp(Node->getOperand(0));
|
|
|
|
SDOperand Op1 = LegalizeOp(Node->getOperand(1));
|
|
|
|
if (Op0 != Node->getOperand(0) || Op1 != Node->getOperand(1))
|
|
|
|
Result = DAG.getNode(ISD::TokenFactor, MVT::Other, Op0, Op1);
|
|
|
|
} else {
|
|
|
|
std::vector<SDOperand> Ops;
|
|
|
|
bool Changed = false;
|
|
|
|
// Legalize the operands.
|
|
|
|
for (unsigned i = 0, e = Node->getNumOperands(); i != e; ++i) {
|
|
|
|
SDOperand Op = Node->getOperand(i);
|
|
|
|
Ops.push_back(LegalizeOp(Op));
|
|
|
|
Changed |= Ops[i] != Op;
|
|
|
|
}
|
|
|
|
if (Changed)
|
|
|
|
Result = DAG.getNode(ISD::TokenFactor, MVT::Other, Ops);
|
2005-01-14 01:59:25 +08:00
|
|
|
}
|
|
|
|
break;
|
|
|
|
|
2005-05-13 07:24:06 +08:00
|
|
|
case ISD::CALLSEQ_START:
|
|
|
|
case ISD::CALLSEQ_END:
|
2005-01-07 15:47:09 +08:00
|
|
|
Tmp1 = LegalizeOp(Node->getOperand(0)); // Legalize the chain.
|
2005-05-13 07:24:44 +08:00
|
|
|
// Do not try to legalize the target-specific arguments (#1+)
|
2005-05-13 00:53:42 +08:00
|
|
|
Tmp2 = Node->getOperand(0);
|
2005-10-04 10:10:55 +08:00
|
|
|
if (Tmp1 != Tmp2)
|
2005-05-12 08:17:04 +08:00
|
|
|
Node->setAdjCallChain(Tmp1);
|
2005-10-04 08:37:37 +08:00
|
|
|
|
2005-05-13 07:24:06 +08:00
|
|
|
// Note that we do not create new CALLSEQ_DOWN/UP nodes here. These
|
2005-05-12 08:17:04 +08:00
|
|
|
// nodes are treated specially and are mutated in place. This makes the dag
|
|
|
|
// legalization process more efficient and also makes libcall insertion
|
|
|
|
// easier.
|
2005-01-07 15:47:09 +08:00
|
|
|
break;
|
2006-01-12 06:14:47 +08:00
|
|
|
case ISD::DYNAMIC_STACKALLOC: {
|
2005-01-10 03:03:49 +08:00
|
|
|
Tmp1 = LegalizeOp(Node->getOperand(0)); // Legalize the chain.
|
|
|
|
Tmp2 = LegalizeOp(Node->getOperand(1)); // Legalize the size.
|
|
|
|
Tmp3 = LegalizeOp(Node->getOperand(2)); // Legalize the alignment.
|
|
|
|
if (Tmp1 != Node->getOperand(0) || Tmp2 != Node->getOperand(1) ||
|
2005-05-14 15:29:57 +08:00
|
|
|
Tmp3 != Node->getOperand(2)) {
|
|
|
|
std::vector<MVT::ValueType> VTs(Node->value_begin(), Node->value_end());
|
|
|
|
std::vector<SDOperand> Ops;
|
|
|
|
Ops.push_back(Tmp1); Ops.push_back(Tmp2); Ops.push_back(Tmp3);
|
|
|
|
Result = DAG.getNode(ISD::DYNAMIC_STACKALLOC, VTs, Ops);
|
|
|
|
} else
|
2005-01-10 03:07:54 +08:00
|
|
|
Result = Op.getValue(0);
|
2005-01-10 03:03:49 +08:00
|
|
|
|
2006-01-15 16:43:08 +08:00
|
|
|
Tmp1 = Result;
|
|
|
|
Tmp2 = Result.getValue(1);
|
2006-01-12 06:14:47 +08:00
|
|
|
switch (TLI.getOperationAction(Node->getOpcode(),
|
|
|
|
Node->getValueType(0))) {
|
|
|
|
default: assert(0 && "This action is not supported yet!");
|
2006-01-15 16:54:32 +08:00
|
|
|
case TargetLowering::Expand: {
|
|
|
|
unsigned SPReg = TLI.getStackPointerRegisterToSaveRestore();
|
|
|
|
assert(SPReg && "Target cannot require DYNAMIC_STACKALLOC expansion and"
|
|
|
|
" not tell us which reg is the stack pointer!");
|
|
|
|
SDOperand Chain = Tmp1.getOperand(0);
|
|
|
|
SDOperand Size = Tmp2.getOperand(1);
|
|
|
|
SDOperand SP = DAG.getCopyFromReg(Chain, SPReg, Node->getValueType(0));
|
|
|
|
Tmp1 = DAG.getNode(ISD::SUB, Node->getValueType(0), SP, Size); // Value
|
|
|
|
Tmp2 = DAG.getCopyToReg(SP.getValue(1), SPReg, Tmp1); // Output chain
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
case TargetLowering::Custom:
|
2006-01-15 16:43:08 +08:00
|
|
|
Tmp3 = TLI.LowerOperation(Tmp1, DAG);
|
|
|
|
if (Tmp3.Val) {
|
|
|
|
Tmp1 = LegalizeOp(Tmp3);
|
|
|
|
Tmp2 = LegalizeOp(Tmp3.getValue(1));
|
2006-01-12 06:14:47 +08:00
|
|
|
}
|
2006-01-15 16:54:32 +08:00
|
|
|
break;
|
2006-01-12 06:14:47 +08:00
|
|
|
case TargetLowering::Legal:
|
2006-01-15 16:54:32 +08:00
|
|
|
break;
|
2006-01-12 06:14:47 +08:00
|
|
|
}
|
2006-01-15 16:54:32 +08:00
|
|
|
// Since this op produce two values, make sure to remember that we
|
|
|
|
// legalized both of them.
|
|
|
|
AddLegalizedOperand(SDOperand(Node, 0), Tmp1);
|
|
|
|
AddLegalizedOperand(SDOperand(Node, 1), Tmp2);
|
|
|
|
return Op.ResNo ? Tmp2 : Tmp1;
|
2006-01-12 06:14:47 +08:00
|
|
|
}
|
2005-05-14 02:43:43 +08:00
|
|
|
case ISD::TAILCALL:
|
2005-01-20 04:24:35 +08:00
|
|
|
case ISD::CALL: {
|
2005-01-07 15:47:09 +08:00
|
|
|
Tmp1 = LegalizeOp(Node->getOperand(0)); // Legalize the chain.
|
|
|
|
Tmp2 = LegalizeOp(Node->getOperand(1)); // Legalize the callee.
|
2005-01-20 04:24:35 +08:00
|
|
|
|
|
|
|
bool Changed = false;
|
|
|
|
std::vector<SDOperand> Ops;
|
|
|
|
for (unsigned i = 2, e = Node->getNumOperands(); i != e; ++i) {
|
|
|
|
Ops.push_back(LegalizeOp(Node->getOperand(i)));
|
|
|
|
Changed |= Ops.back() != Node->getOperand(i);
|
|
|
|
}
|
|
|
|
|
|
|
|
if (Tmp1 != Node->getOperand(0) || Tmp2 != Node->getOperand(1) || Changed) {
|
2005-01-07 15:47:09 +08:00
|
|
|
std::vector<MVT::ValueType> RetTyVTs;
|
|
|
|
RetTyVTs.reserve(Node->getNumValues());
|
|
|
|
for (unsigned i = 0, e = Node->getNumValues(); i != e; ++i)
|
2005-01-08 05:34:13 +08:00
|
|
|
RetTyVTs.push_back(Node->getValueType(i));
|
2005-05-14 02:43:43 +08:00
|
|
|
Result = SDOperand(DAG.getCall(RetTyVTs, Tmp1, Tmp2, Ops,
|
|
|
|
Node->getOpcode() == ISD::TAILCALL), 0);
|
2005-01-10 03:43:23 +08:00
|
|
|
} else {
|
|
|
|
Result = Result.getValue(0);
|
2005-01-07 15:47:09 +08:00
|
|
|
}
|
2005-01-10 03:43:23 +08:00
|
|
|
// Since calls produce multiple values, make sure to remember that we
|
|
|
|
// legalized all of them.
|
|
|
|
for (unsigned i = 0, e = Node->getNumValues(); i != e; ++i)
|
|
|
|
AddLegalizedOperand(SDOperand(Node, i), Result.getValue(i));
|
|
|
|
return Result.getValue(Op.ResNo);
|
2005-01-20 04:24:35 +08:00
|
|
|
}
|
2005-01-08 06:12:08 +08:00
|
|
|
case ISD::BR:
|
|
|
|
Tmp1 = LegalizeOp(Node->getOperand(0)); // Legalize the chain.
|
|
|
|
if (Tmp1 != Node->getOperand(0))
|
|
|
|
Result = DAG.getNode(ISD::BR, MVT::Other, Tmp1, Node->getOperand(1));
|
|
|
|
break;
|
|
|
|
|
2005-01-07 16:19:42 +08:00
|
|
|
case ISD::BRCOND:
|
|
|
|
Tmp1 = LegalizeOp(Node->getOperand(0)); // Legalize the chain.
|
2005-08-17 03:49:35 +08:00
|
|
|
|
2005-01-19 03:27:06 +08:00
|
|
|
switch (getTypeAction(Node->getOperand(1).getValueType())) {
|
|
|
|
case Expand: assert(0 && "It's impossible to expand bools");
|
|
|
|
case Legal:
|
|
|
|
Tmp2 = LegalizeOp(Node->getOperand(1)); // Legalize the condition.
|
|
|
|
break;
|
|
|
|
case Promote:
|
|
|
|
Tmp2 = PromoteOp(Node->getOperand(1)); // Promote the condition.
|
|
|
|
break;
|
|
|
|
}
|
2005-08-17 03:49:35 +08:00
|
|
|
|
|
|
|
switch (TLI.getOperationAction(ISD::BRCOND, MVT::Other)) {
|
|
|
|
default: assert(0 && "This action is not supported yet!");
|
|
|
|
case TargetLowering::Expand:
|
|
|
|
// Expand brcond's setcc into its constituent parts and create a BR_CC
|
|
|
|
// Node.
|
|
|
|
if (Tmp2.getOpcode() == ISD::SETCC) {
|
|
|
|
Result = DAG.getNode(ISD::BR_CC, MVT::Other, Tmp1, Tmp2.getOperand(2),
|
|
|
|
Tmp2.getOperand(0), Tmp2.getOperand(1),
|
|
|
|
Node->getOperand(2));
|
|
|
|
} else {
|
When legalizing brcond ->brcc or select -> selectcc, make sure to truncate
the old condition to a one bit value. The incoming value must have been
promoted, and the top bits are undefined. This causes us to generate:
_test:
rlwinm r2, r3, 0, 31, 31
li r3, 17
cmpwi cr0, r2, 0
bne .LBB_test_2 ;
.LBB_test_1: ;
li r3, 1
.LBB_test_2: ;
blr
instead of:
_test:
rlwinm r2, r3, 0, 31, 31
li r2, 17
cmpwi cr0, r3, 0
bne .LBB_test_2 ;
.LBB_test_1: ;
li r2, 1
.LBB_test_2: ;
or r3, r2, r2
blr
for:
int %test(bool %c) {
%retval = select bool %c, int 17, int 1
ret int %retval
}
llvm-svn: 22947
2005-08-22 02:03:09 +08:00
|
|
|
// Make sure the condition is either zero or one. It may have been
|
|
|
|
// promoted from something else.
|
|
|
|
Tmp2 = DAG.getZeroExtendInReg(Tmp2, MVT::i1);
|
|
|
|
|
2005-08-17 03:49:35 +08:00
|
|
|
Result = DAG.getNode(ISD::BR_CC, MVT::Other, Tmp1,
|
|
|
|
DAG.getCondCode(ISD::SETNE), Tmp2,
|
|
|
|
DAG.getConstant(0, Tmp2.getValueType()),
|
|
|
|
Node->getOperand(2));
|
|
|
|
}
|
2005-12-19 07:54:29 +08:00
|
|
|
Result = LegalizeOp(Result); // Relegalize new nodes.
|
2005-08-17 03:49:35 +08:00
|
|
|
break;
|
2005-12-20 07:12:38 +08:00
|
|
|
case TargetLowering::Custom: {
|
|
|
|
SDOperand Tmp =
|
|
|
|
TLI.LowerOperation(DAG.getNode(ISD::BRCOND, Node->getValueType(0),
|
|
|
|
Tmp1, Tmp2, Node->getOperand(2)), DAG);
|
|
|
|
if (Tmp.Val) {
|
|
|
|
Result = LegalizeOp(Tmp);
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
// FALLTHROUGH if the target thinks it is legal.
|
|
|
|
}
|
2005-08-17 03:49:35 +08:00
|
|
|
case TargetLowering::Legal:
|
|
|
|
// Basic block destination (Op#2) is always legal.
|
|
|
|
if (Tmp1 != Node->getOperand(0) || Tmp2 != Node->getOperand(1))
|
|
|
|
Result = DAG.getNode(ISD::BRCOND, MVT::Other, Tmp1, Tmp2,
|
|
|
|
Node->getOperand(2));
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
case ISD::BR_CC:
|
|
|
|
Tmp1 = LegalizeOp(Node->getOperand(0)); // Legalize the chain.
|
2005-12-18 07:46:46 +08:00
|
|
|
if (!isTypeLegal(Node->getOperand(2).getValueType())) {
|
2005-08-17 03:49:35 +08:00
|
|
|
Tmp2 = LegalizeOp(DAG.getNode(ISD::SETCC, TLI.getSetCCResultTy(),
|
|
|
|
Node->getOperand(2), // LHS
|
|
|
|
Node->getOperand(3), // RHS
|
|
|
|
Node->getOperand(1)));
|
|
|
|
// If we get a SETCC back from legalizing the SETCC node we just
|
|
|
|
// created, then use its LHS, RHS, and CC directly in creating a new
|
|
|
|
// node. Otherwise, select between the true and false value based on
|
|
|
|
// comparing the result of the legalized with zero.
|
|
|
|
if (Tmp2.getOpcode() == ISD::SETCC) {
|
|
|
|
Result = DAG.getNode(ISD::BR_CC, MVT::Other, Tmp1, Tmp2.getOperand(2),
|
|
|
|
Tmp2.getOperand(0), Tmp2.getOperand(1),
|
|
|
|
Node->getOperand(4));
|
|
|
|
} else {
|
|
|
|
Result = DAG.getNode(ISD::BR_CC, MVT::Other, Tmp1,
|
|
|
|
DAG.getCondCode(ISD::SETNE),
|
|
|
|
Tmp2, DAG.getConstant(0, Tmp2.getValueType()),
|
|
|
|
Node->getOperand(4));
|
|
|
|
}
|
2005-12-18 07:46:46 +08:00
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
|
|
|
Tmp2 = LegalizeOp(Node->getOperand(2)); // LHS
|
|
|
|
Tmp3 = LegalizeOp(Node->getOperand(3)); // RHS
|
|
|
|
|
|
|
|
switch (TLI.getOperationAction(ISD::BR_CC, Tmp3.getValueType())) {
|
|
|
|
default: assert(0 && "Unexpected action for BR_CC!");
|
|
|
|
case TargetLowering::Custom: {
|
|
|
|
Tmp4 = DAG.getNode(ISD::BR_CC, MVT::Other, Tmp1, Node->getOperand(1),
|
|
|
|
Tmp2, Tmp3, Node->getOperand(4));
|
|
|
|
Tmp4 = TLI.LowerOperation(Tmp4, DAG);
|
|
|
|
if (Tmp4.Val) {
|
|
|
|
Result = LegalizeOp(Tmp4);
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
} // FALLTHROUGH if the target doesn't want to lower this op after all.
|
|
|
|
case TargetLowering::Legal:
|
|
|
|
if (Tmp1 != Node->getOperand(0) || Tmp2 != Node->getOperand(2) ||
|
|
|
|
Tmp3 != Node->getOperand(3)) {
|
|
|
|
Result = DAG.getNode(ISD::BR_CC, MVT::Other, Tmp1, Node->getOperand(1),
|
|
|
|
Tmp2, Tmp3, Node->getOperand(4));
|
|
|
|
}
|
|
|
|
break;
|
2005-08-17 03:49:35 +08:00
|
|
|
}
|
2005-01-07 16:19:42 +08:00
|
|
|
break;
|
2005-04-09 11:30:19 +08:00
|
|
|
case ISD::BRCONDTWOWAY:
|
|
|
|
Tmp1 = LegalizeOp(Node->getOperand(0)); // Legalize the chain.
|
|
|
|
switch (getTypeAction(Node->getOperand(1).getValueType())) {
|
|
|
|
case Expand: assert(0 && "It's impossible to expand bools");
|
|
|
|
case Legal:
|
|
|
|
Tmp2 = LegalizeOp(Node->getOperand(1)); // Legalize the condition.
|
|
|
|
break;
|
|
|
|
case Promote:
|
|
|
|
Tmp2 = PromoteOp(Node->getOperand(1)); // Promote the condition.
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
// If this target does not support BRCONDTWOWAY, lower it to a BRCOND/BR
|
|
|
|
// pair.
|
|
|
|
switch (TLI.getOperationAction(ISD::BRCONDTWOWAY, MVT::Other)) {
|
|
|
|
case TargetLowering::Promote:
|
|
|
|
default: assert(0 && "This action is not supported yet!");
|
|
|
|
case TargetLowering::Legal:
|
|
|
|
if (Tmp1 != Node->getOperand(0) || Tmp2 != Node->getOperand(1)) {
|
|
|
|
std::vector<SDOperand> Ops;
|
|
|
|
Ops.push_back(Tmp1);
|
|
|
|
Ops.push_back(Tmp2);
|
|
|
|
Ops.push_back(Node->getOperand(2));
|
|
|
|
Ops.push_back(Node->getOperand(3));
|
|
|
|
Result = DAG.getNode(ISD::BRCONDTWOWAY, MVT::Other, Ops);
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
case TargetLowering::Expand:
|
2005-08-17 03:49:35 +08:00
|
|
|
// If BRTWOWAY_CC is legal for this target, then simply expand this node
|
|
|
|
// to that. Otherwise, skip BRTWOWAY_CC and expand directly to a
|
|
|
|
// BRCOND/BR pair.
|
2005-08-25 00:35:28 +08:00
|
|
|
if (TLI.isOperationLegal(ISD::BRTWOWAY_CC, MVT::Other)) {
|
2005-08-17 03:49:35 +08:00
|
|
|
if (Tmp2.getOpcode() == ISD::SETCC) {
|
|
|
|
Result = DAG.getBR2Way_CC(Tmp1, Tmp2.getOperand(2),
|
|
|
|
Tmp2.getOperand(0), Tmp2.getOperand(1),
|
|
|
|
Node->getOperand(2), Node->getOperand(3));
|
|
|
|
} else {
|
|
|
|
Result = DAG.getBR2Way_CC(Tmp1, DAG.getCondCode(ISD::SETNE), Tmp2,
|
|
|
|
DAG.getConstant(0, Tmp2.getValueType()),
|
|
|
|
Node->getOperand(2), Node->getOperand(3));
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
Result = DAG.getNode(ISD::BRCOND, MVT::Other, Tmp1, Tmp2,
|
2005-04-09 11:30:19 +08:00
|
|
|
Node->getOperand(2));
|
2005-08-17 03:49:35 +08:00
|
|
|
Result = DAG.getNode(ISD::BR, MVT::Other, Result, Node->getOperand(3));
|
|
|
|
}
|
2005-12-19 07:54:29 +08:00
|
|
|
Result = LegalizeOp(Result); // Relegalize new nodes.
|
2005-04-09 11:30:19 +08:00
|
|
|
break;
|
|
|
|
}
|
|
|
|
break;
|
2005-08-17 03:49:35 +08:00
|
|
|
case ISD::BRTWOWAY_CC:
|
|
|
|
Tmp1 = LegalizeOp(Node->getOperand(0)); // Legalize the chain.
|
2005-08-25 00:35:28 +08:00
|
|
|
if (isTypeLegal(Node->getOperand(2).getValueType())) {
|
2005-08-17 03:49:35 +08:00
|
|
|
Tmp2 = LegalizeOp(Node->getOperand(2)); // LHS
|
|
|
|
Tmp3 = LegalizeOp(Node->getOperand(3)); // RHS
|
|
|
|
if (Tmp1 != Node->getOperand(0) || Tmp2 != Node->getOperand(2) ||
|
|
|
|
Tmp3 != Node->getOperand(3)) {
|
|
|
|
Result = DAG.getBR2Way_CC(Tmp1, Node->getOperand(1), Tmp2, Tmp3,
|
|
|
|
Node->getOperand(4), Node->getOperand(5));
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
} else {
|
|
|
|
Tmp2 = LegalizeOp(DAG.getNode(ISD::SETCC, TLI.getSetCCResultTy(),
|
|
|
|
Node->getOperand(2), // LHS
|
|
|
|
Node->getOperand(3), // RHS
|
|
|
|
Node->getOperand(1)));
|
|
|
|
// If this target does not support BRTWOWAY_CC, lower it to a BRCOND/BR
|
|
|
|
// pair.
|
|
|
|
switch (TLI.getOperationAction(ISD::BRTWOWAY_CC, MVT::Other)) {
|
|
|
|
default: assert(0 && "This action is not supported yet!");
|
|
|
|
case TargetLowering::Legal:
|
|
|
|
// If we get a SETCC back from legalizing the SETCC node we just
|
|
|
|
// created, then use its LHS, RHS, and CC directly in creating a new
|
|
|
|
// node. Otherwise, select between the true and false value based on
|
|
|
|
// comparing the result of the legalized with zero.
|
|
|
|
if (Tmp2.getOpcode() == ISD::SETCC) {
|
|
|
|
Result = DAG.getBR2Way_CC(Tmp1, Tmp2.getOperand(2),
|
|
|
|
Tmp2.getOperand(0), Tmp2.getOperand(1),
|
|
|
|
Node->getOperand(4), Node->getOperand(5));
|
|
|
|
} else {
|
|
|
|
Result = DAG.getBR2Way_CC(Tmp1, DAG.getCondCode(ISD::SETNE), Tmp2,
|
|
|
|
DAG.getConstant(0, Tmp2.getValueType()),
|
|
|
|
Node->getOperand(4), Node->getOperand(5));
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
case TargetLowering::Expand:
|
|
|
|
Result = DAG.getNode(ISD::BRCOND, MVT::Other, Tmp1, Tmp2,
|
|
|
|
Node->getOperand(4));
|
|
|
|
Result = DAG.getNode(ISD::BR, MVT::Other, Result, Node->getOperand(5));
|
|
|
|
break;
|
|
|
|
}
|
2005-12-22 03:40:42 +08:00
|
|
|
Result = LegalizeOp(Result); // Relegalize new nodes.
|
2005-08-17 03:49:35 +08:00
|
|
|
}
|
|
|
|
break;
|
2005-12-23 15:29:34 +08:00
|
|
|
case ISD::LOAD: {
|
2005-01-07 15:47:09 +08:00
|
|
|
Tmp1 = LegalizeOp(Node->getOperand(0)); // Legalize the chain.
|
|
|
|
Tmp2 = LegalizeOp(Node->getOperand(1)); // Legalize the pointer.
|
2005-04-28 04:10:01 +08:00
|
|
|
|
2005-12-23 15:29:34 +08:00
|
|
|
MVT::ValueType VT = Node->getValueType(0);
|
|
|
|
switch (TLI.getOperationAction(Node->getOpcode(), VT)) {
|
|
|
|
default: assert(0 && "This action is not supported yet!");
|
|
|
|
case TargetLowering::Custom: {
|
|
|
|
SDOperand Op = DAG.getLoad(Node->getValueType(0),
|
|
|
|
Tmp1, Tmp2, Node->getOperand(2));
|
|
|
|
SDOperand Tmp = TLI.LowerOperation(Op, DAG);
|
|
|
|
if (Tmp.Val) {
|
|
|
|
Result = LegalizeOp(Tmp);
|
|
|
|
// Since loads produce two values, make sure to remember that we legalized
|
|
|
|
// both of them.
|
|
|
|
AddLegalizedOperand(SDOperand(Node, 0), Result);
|
|
|
|
AddLegalizedOperand(SDOperand(Node, 1), Result.getValue(1));
|
|
|
|
return Result.getValue(Op.ResNo);
|
|
|
|
}
|
|
|
|
// FALLTHROUGH if the target thinks it is legal.
|
|
|
|
}
|
|
|
|
case TargetLowering::Legal:
|
|
|
|
if (Tmp1 != Node->getOperand(0) ||
|
|
|
|
Tmp2 != Node->getOperand(1))
|
|
|
|
Result = DAG.getLoad(Node->getValueType(0), Tmp1, Tmp2,
|
|
|
|
Node->getOperand(2));
|
|
|
|
else
|
|
|
|
Result = SDOperand(Node, 0);
|
2005-04-22 06:36:52 +08:00
|
|
|
|
2005-12-23 15:29:34 +08:00
|
|
|
// Since loads produce two values, make sure to remember that we legalized
|
|
|
|
// both of them.
|
|
|
|
AddLegalizedOperand(SDOperand(Node, 0), Result);
|
|
|
|
AddLegalizedOperand(SDOperand(Node, 1), Result.getValue(1));
|
|
|
|
return Result.getValue(Op.ResNo);
|
|
|
|
}
|
|
|
|
assert(0 && "Unreachable");
|
|
|
|
}
|
2005-01-15 14:18:18 +08:00
|
|
|
case ISD::EXTLOAD:
|
|
|
|
case ISD::SEXTLOAD:
|
2005-04-11 06:54:25 +08:00
|
|
|
case ISD::ZEXTLOAD: {
|
2005-01-15 14:18:18 +08:00
|
|
|
Tmp1 = LegalizeOp(Node->getOperand(0)); // Legalize the chain.
|
|
|
|
Tmp2 = LegalizeOp(Node->getOperand(1)); // Legalize the pointer.
|
|
|
|
|
2005-07-10 09:55:33 +08:00
|
|
|
MVT::ValueType SrcVT = cast<VTSDNode>(Node->getOperand(3))->getVT();
|
2005-04-11 06:54:25 +08:00
|
|
|
switch (TLI.getOperationAction(Node->getOpcode(), SrcVT)) {
|
|
|
|
default: assert(0 && "This action is not supported yet!");
|
2005-04-13 04:30:10 +08:00
|
|
|
case TargetLowering::Promote:
|
|
|
|
assert(SrcVT == MVT::i1 && "Can only promote EXTLOAD from i1 -> i8!");
|
2005-07-10 09:55:33 +08:00
|
|
|
Result = DAG.getExtLoad(Node->getOpcode(), Node->getValueType(0),
|
|
|
|
Tmp1, Tmp2, Node->getOperand(2), MVT::i8);
|
2005-04-13 04:30:10 +08:00
|
|
|
// Since loads produce two values, make sure to remember that we legalized
|
|
|
|
// both of them.
|
|
|
|
AddLegalizedOperand(SDOperand(Node, 0), Result);
|
|
|
|
AddLegalizedOperand(SDOperand(Node, 1), Result.getValue(1));
|
|
|
|
return Result.getValue(Op.ResNo);
|
2005-04-22 06:36:52 +08:00
|
|
|
|
2005-12-23 15:29:34 +08:00
|
|
|
case TargetLowering::Custom: {
|
|
|
|
SDOperand Op = DAG.getExtLoad(Node->getOpcode(), Node->getValueType(0),
|
|
|
|
Tmp1, Tmp2, Node->getOperand(2),
|
|
|
|
SrcVT);
|
|
|
|
SDOperand Tmp = TLI.LowerOperation(Op, DAG);
|
|
|
|
if (Tmp.Val) {
|
|
|
|
Result = LegalizeOp(Tmp);
|
|
|
|
// Since loads produce two values, make sure to remember that we legalized
|
|
|
|
// both of them.
|
|
|
|
AddLegalizedOperand(SDOperand(Node, 0), Result);
|
|
|
|
AddLegalizedOperand(SDOperand(Node, 1), Result.getValue(1));
|
|
|
|
return Result.getValue(Op.ResNo);
|
|
|
|
}
|
|
|
|
// FALLTHROUGH if the target thinks it is legal.
|
|
|
|
}
|
2005-04-11 06:54:25 +08:00
|
|
|
case TargetLowering::Legal:
|
|
|
|
if (Tmp1 != Node->getOperand(0) ||
|
|
|
|
Tmp2 != Node->getOperand(1))
|
2005-07-10 09:55:33 +08:00
|
|
|
Result = DAG.getExtLoad(Node->getOpcode(), Node->getValueType(0),
|
|
|
|
Tmp1, Tmp2, Node->getOperand(2), SrcVT);
|
2005-04-11 06:54:25 +08:00
|
|
|
else
|
|
|
|
Result = SDOperand(Node, 0);
|
|
|
|
|
|
|
|
// Since loads produce two values, make sure to remember that we legalized
|
|
|
|
// both of them.
|
|
|
|
AddLegalizedOperand(SDOperand(Node, 0), Result);
|
|
|
|
AddLegalizedOperand(SDOperand(Node, 1), Result.getValue(1));
|
|
|
|
return Result.getValue(Op.ResNo);
|
|
|
|
case TargetLowering::Expand:
|
2005-12-20 08:53:54 +08:00
|
|
|
// f64 = EXTLOAD f32 should expand to LOAD, FP_EXTEND
|
2005-07-01 03:22:37 +08:00
|
|
|
if (SrcVT == MVT::f32 && Node->getValueType(0) == MVT::f64) {
|
|
|
|
SDOperand Load = DAG.getLoad(SrcVT, Tmp1, Tmp2, Node->getOperand(2));
|
2005-07-01 03:32:57 +08:00
|
|
|
Result = DAG.getNode(ISD::FP_EXTEND, Node->getValueType(0), Load);
|
2005-12-19 07:54:29 +08:00
|
|
|
Result = LegalizeOp(Result); // Relegalize new nodes.
|
2005-12-20 08:53:54 +08:00
|
|
|
Load = LegalizeOp(Load);
|
|
|
|
AddLegalizedOperand(SDOperand(Node, 0), Result);
|
|
|
|
AddLegalizedOperand(SDOperand(Node, 1), Load.getValue(1));
|
2005-07-01 03:22:37 +08:00
|
|
|
if (Op.ResNo)
|
|
|
|
return Load.getValue(1);
|
|
|
|
return Result;
|
|
|
|
}
|
2005-04-11 06:54:25 +08:00
|
|
|
assert(Node->getOpcode() != ISD::EXTLOAD &&
|
|
|
|
"EXTLOAD should always be supported!");
|
|
|
|
// Turn the unsupported load into an EXTLOAD followed by an explicit
|
|
|
|
// zero/sign extend inreg.
|
2005-07-10 09:55:33 +08:00
|
|
|
Result = DAG.getExtLoad(ISD::EXTLOAD, Node->getValueType(0),
|
|
|
|
Tmp1, Tmp2, Node->getOperand(2), SrcVT);
|
2005-04-13 10:38:47 +08:00
|
|
|
SDOperand ValRes;
|
|
|
|
if (Node->getOpcode() == ISD::SEXTLOAD)
|
|
|
|
ValRes = DAG.getNode(ISD::SIGN_EXTEND_INREG, Result.getValueType(),
|
2005-07-10 08:07:11 +08:00
|
|
|
Result, DAG.getValueType(SrcVT));
|
2005-04-13 10:38:47 +08:00
|
|
|
else
|
|
|
|
ValRes = DAG.getZeroExtendInReg(Result, SrcVT);
|
2005-12-19 07:54:29 +08:00
|
|
|
Result = LegalizeOp(Result); // Relegalize new nodes.
|
|
|
|
ValRes = LegalizeOp(ValRes); // Relegalize new nodes.
|
2005-12-20 08:53:54 +08:00
|
|
|
AddLegalizedOperand(SDOperand(Node, 0), ValRes);
|
|
|
|
AddLegalizedOperand(SDOperand(Node, 1), Result.getValue(1));
|
2005-04-11 06:54:25 +08:00
|
|
|
if (Op.ResNo)
|
|
|
|
return Result.getValue(1);
|
|
|
|
return ValRes;
|
|
|
|
}
|
|
|
|
assert(0 && "Unreachable");
|
|
|
|
}
|
2005-10-19 08:06:56 +08:00
|
|
|
case ISD::EXTRACT_ELEMENT: {
|
|
|
|
MVT::ValueType OpTy = Node->getOperand(0).getValueType();
|
|
|
|
switch (getTypeAction(OpTy)) {
|
|
|
|
default:
|
|
|
|
assert(0 && "EXTRACT_ELEMENT action for type unimplemented!");
|
|
|
|
break;
|
|
|
|
case Legal:
|
|
|
|
if (cast<ConstantSDNode>(Node->getOperand(1))->getValue()) {
|
|
|
|
// 1 -> Hi
|
|
|
|
Result = DAG.getNode(ISD::SRL, OpTy, Node->getOperand(0),
|
|
|
|
DAG.getConstant(MVT::getSizeInBits(OpTy)/2,
|
|
|
|
TLI.getShiftAmountTy()));
|
|
|
|
Result = DAG.getNode(ISD::TRUNCATE, Node->getValueType(0), Result);
|
|
|
|
} else {
|
|
|
|
// 0 -> Lo
|
|
|
|
Result = DAG.getNode(ISD::TRUNCATE, Node->getValueType(0),
|
|
|
|
Node->getOperand(0));
|
|
|
|
}
|
|
|
|
Result = LegalizeOp(Result);
|
|
|
|
break;
|
|
|
|
case Expand:
|
|
|
|
// Get both the low and high parts.
|
|
|
|
ExpandOp(Node->getOperand(0), Tmp1, Tmp2);
|
|
|
|
if (cast<ConstantSDNode>(Node->getOperand(1))->getValue())
|
|
|
|
Result = Tmp2; // 1 -> Hi
|
|
|
|
else
|
|
|
|
Result = Tmp1; // 0 -> Lo
|
|
|
|
break;
|
|
|
|
}
|
2005-01-07 15:47:09 +08:00
|
|
|
break;
|
2005-10-19 08:06:56 +08:00
|
|
|
}
|
2005-01-07 15:47:09 +08:00
|
|
|
|
|
|
|
case ISD::CopyToReg:
|
|
|
|
Tmp1 = LegalizeOp(Node->getOperand(0)); // Legalize the chain.
|
2005-04-22 06:36:52 +08:00
|
|
|
|
2005-08-25 00:35:28 +08:00
|
|
|
assert(isTypeLegal(Node->getOperand(2).getValueType()) &&
|
2005-08-17 05:55:35 +08:00
|
|
|
"Register type must be legal!");
|
2005-12-18 23:27:43 +08:00
|
|
|
// Legalize the incoming value (must be a legal type).
|
2005-08-17 05:55:35 +08:00
|
|
|
Tmp2 = LegalizeOp(Node->getOperand(2));
|
2005-12-18 23:36:21 +08:00
|
|
|
if (Node->getNumValues() == 1) {
|
2005-12-18 23:27:43 +08:00
|
|
|
if (Tmp1 != Node->getOperand(0) || Tmp2 != Node->getOperand(2))
|
|
|
|
Result = DAG.getNode(ISD::CopyToReg, MVT::Other, Tmp1,
|
|
|
|
Node->getOperand(1), Tmp2);
|
|
|
|
} else {
|
2005-12-18 23:36:21 +08:00
|
|
|
assert(Node->getNumValues() == 2 && "Unknown CopyToReg");
|
|
|
|
if (Node->getNumOperands() == 4)
|
|
|
|
Tmp3 = LegalizeOp(Node->getOperand(3));
|
2005-12-18 23:27:43 +08:00
|
|
|
if (Tmp1 != Node->getOperand(0) || Tmp2 != Node->getOperand(2) ||
|
2005-12-18 23:36:21 +08:00
|
|
|
(Node->getNumOperands() == 4 && Tmp3 != Node->getOperand(3))) {
|
2005-12-18 23:27:43 +08:00
|
|
|
unsigned Reg = cast<RegisterSDNode>(Node->getOperand(1))->getReg();
|
|
|
|
Result = DAG.getCopyToReg(Tmp1, Reg, Tmp2, Tmp3);
|
|
|
|
}
|
|
|
|
|
|
|
|
// Since this produces two values, make sure to remember that we legalized
|
|
|
|
// both of them.
|
|
|
|
AddLegalizedOperand(SDOperand(Node, 0), Result);
|
|
|
|
AddLegalizedOperand(SDOperand(Node, 1), Result.getValue(1));
|
|
|
|
return Result.getValue(Op.ResNo);
|
|
|
|
}
|
2005-01-07 15:47:09 +08:00
|
|
|
break;
|
|
|
|
|
|
|
|
case ISD::RET:
|
|
|
|
Tmp1 = LegalizeOp(Node->getOperand(0)); // Legalize the chain.
|
|
|
|
switch (Node->getNumOperands()) {
|
|
|
|
case 2: // ret val
|
|
|
|
switch (getTypeAction(Node->getOperand(1).getValueType())) {
|
|
|
|
case Legal:
|
|
|
|
Tmp2 = LegalizeOp(Node->getOperand(1));
|
2005-01-08 06:28:47 +08:00
|
|
|
if (Tmp1 != Node->getOperand(0) || Tmp2 != Node->getOperand(1))
|
2005-01-07 15:47:09 +08:00
|
|
|
Result = DAG.getNode(ISD::RET, MVT::Other, Tmp1, Tmp2);
|
|
|
|
break;
|
|
|
|
case Expand: {
|
|
|
|
SDOperand Lo, Hi;
|
|
|
|
ExpandOp(Node->getOperand(1), Lo, Hi);
|
|
|
|
Result = DAG.getNode(ISD::RET, MVT::Other, Tmp1, Lo, Hi);
|
2005-04-22 06:36:52 +08:00
|
|
|
break;
|
2005-01-07 15:47:09 +08:00
|
|
|
}
|
|
|
|
case Promote:
|
2005-01-16 06:16:26 +08:00
|
|
|
Tmp2 = PromoteOp(Node->getOperand(1));
|
|
|
|
Result = DAG.getNode(ISD::RET, MVT::Other, Tmp1, Tmp2);
|
|
|
|
break;
|
2005-01-07 15:47:09 +08:00
|
|
|
}
|
|
|
|
break;
|
|
|
|
case 1: // ret void
|
|
|
|
if (Tmp1 != Node->getOperand(0))
|
|
|
|
Result = DAG.getNode(ISD::RET, MVT::Other, Tmp1);
|
|
|
|
break;
|
|
|
|
default: { // ret <values>
|
|
|
|
std::vector<SDOperand> NewValues;
|
|
|
|
NewValues.push_back(Tmp1);
|
|
|
|
for (unsigned i = 1, e = Node->getNumOperands(); i != e; ++i)
|
|
|
|
switch (getTypeAction(Node->getOperand(i).getValueType())) {
|
|
|
|
case Legal:
|
2005-01-09 03:27:05 +08:00
|
|
|
NewValues.push_back(LegalizeOp(Node->getOperand(i)));
|
2005-01-07 15:47:09 +08:00
|
|
|
break;
|
|
|
|
case Expand: {
|
|
|
|
SDOperand Lo, Hi;
|
|
|
|
ExpandOp(Node->getOperand(i), Lo, Hi);
|
|
|
|
NewValues.push_back(Lo);
|
|
|
|
NewValues.push_back(Hi);
|
2005-04-22 06:36:52 +08:00
|
|
|
break;
|
2005-01-07 15:47:09 +08:00
|
|
|
}
|
|
|
|
case Promote:
|
2005-01-16 06:16:26 +08:00
|
|
|
assert(0 && "Can't promote multiple return value yet!");
|
2005-01-07 15:47:09 +08:00
|
|
|
}
|
|
|
|
Result = DAG.getNode(ISD::RET, MVT::Other, NewValues);
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
2006-01-06 08:41:43 +08:00
|
|
|
|
2006-01-06 13:47:48 +08:00
|
|
|
switch (TLI.getOperationAction(Node->getOpcode(),
|
|
|
|
Node->getValueType(0))) {
|
2006-01-06 08:41:43 +08:00
|
|
|
default: assert(0 && "This action is not supported yet!");
|
|
|
|
case TargetLowering::Custom: {
|
|
|
|
SDOperand Tmp = TLI.LowerOperation(Result, DAG);
|
|
|
|
if (Tmp.Val) {
|
|
|
|
Result = LegalizeOp(Tmp);
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
// FALLTHROUGH if the target thinks it is legal.
|
|
|
|
}
|
|
|
|
case TargetLowering::Legal:
|
|
|
|
// Nothing to do.
|
|
|
|
break;
|
|
|
|
}
|
2005-01-07 15:47:09 +08:00
|
|
|
break;
|
2005-12-23 15:29:34 +08:00
|
|
|
case ISD::STORE: {
|
2005-01-07 15:47:09 +08:00
|
|
|
Tmp1 = LegalizeOp(Node->getOperand(0)); // Legalize the chain.
|
|
|
|
Tmp2 = LegalizeOp(Node->getOperand(2)); // Legalize the pointer.
|
|
|
|
|
2005-01-08 14:25:56 +08:00
|
|
|
// Turn 'store float 1.0, Ptr' -> 'store int 0x12345678, Ptr'
|
2005-01-15 13:21:40 +08:00
|
|
|
if (ConstantFPSDNode *CFP =dyn_cast<ConstantFPSDNode>(Node->getOperand(1))){
|
2005-01-08 14:25:56 +08:00
|
|
|
if (CFP->getValueType(0) == MVT::f32) {
|
2005-07-27 14:12:32 +08:00
|
|
|
Result = DAG.getNode(ISD::STORE, MVT::Other, Tmp1,
|
2005-08-18 03:34:49 +08:00
|
|
|
DAG.getConstant(FloatToBits(CFP->getValue()),
|
|
|
|
MVT::i32),
|
|
|
|
Tmp2,
|
2005-05-10 04:23:03 +08:00
|
|
|
Node->getOperand(3));
|
2005-01-08 14:25:56 +08:00
|
|
|
} else {
|
|
|
|
assert(CFP->getValueType(0) == MVT::f64 && "Unknown FP type!");
|
2005-07-27 14:12:32 +08:00
|
|
|
Result = DAG.getNode(ISD::STORE, MVT::Other, Tmp1,
|
2005-08-18 03:34:49 +08:00
|
|
|
DAG.getConstant(DoubleToBits(CFP->getValue()),
|
|
|
|
MVT::i64),
|
|
|
|
Tmp2,
|
2005-05-10 04:23:03 +08:00
|
|
|
Node->getOperand(3));
|
2005-01-08 14:25:56 +08:00
|
|
|
}
|
Fix a bug in the 'store fpimm, ptr' -> 'store intimm, ptr' handling code.
Changing 'op' here caused us to not enter the store into a map, causing
reemission of the code!! In practice, a simple loop like this:
no_exit: ; preds = %no_exit, %entry
%indvar = phi uint [ %indvar.next, %no_exit ], [ 0, %entry ] ; <uint> [#uses=3]
%tmp.4 = getelementptr "complex long double"* %P, uint %indvar, uint 0 ; <double*> [#uses=1]
store double 0.000000e+00, double* %tmp.4
%indvar.next = add uint %indvar, 1 ; <uint> [#uses=2]
%exitcond = seteq uint %indvar.next, %N ; <bool> [#uses=1]
br bool %exitcond, label %return, label %no_exit
was being code gen'd to:
.LBBtest_1: # no_exit
movl %edx, %esi
shll $4, %esi
movl $0, 4(%eax,%esi)
movl $0, (%eax,%esi)
incl %edx
movl $0, (%eax,%esi)
movl $0, 4(%eax,%esi)
cmpl %ecx, %edx
jne .LBBtest_1 # no_exit
Note that we are doing 4 32-bit stores instead of 2. Now we generate:
.LBBtest_1: # no_exit
movl %edx, %esi
incl %esi
shll $4, %edx
movl $0, (%eax,%edx)
movl $0, 4(%eax,%edx)
cmpl %ecx, %esi
movl %esi, %edx
jne .LBBtest_1 # no_exit
This is much happier, though it would be even better if the increment of ESI
was scheduled after the compare :-/
llvm-svn: 20265
2005-02-22 15:23:39 +08:00
|
|
|
Node = Result.Val;
|
2005-01-08 14:25:56 +08:00
|
|
|
}
|
|
|
|
|
2005-01-07 15:47:09 +08:00
|
|
|
switch (getTypeAction(Node->getOperand(1).getValueType())) {
|
|
|
|
case Legal: {
|
|
|
|
SDOperand Val = LegalizeOp(Node->getOperand(1));
|
|
|
|
if (Val != Node->getOperand(1) || Tmp1 != Node->getOperand(0) ||
|
|
|
|
Tmp2 != Node->getOperand(2))
|
2005-05-10 04:23:03 +08:00
|
|
|
Result = DAG.getNode(ISD::STORE, MVT::Other, Tmp1, Val, Tmp2,
|
|
|
|
Node->getOperand(3));
|
2005-12-23 15:29:34 +08:00
|
|
|
|
|
|
|
MVT::ValueType VT = Result.Val->getOperand(1).getValueType();
|
|
|
|
switch (TLI.getOperationAction(Result.Val->getOpcode(), VT)) {
|
|
|
|
default: assert(0 && "This action is not supported yet!");
|
|
|
|
case TargetLowering::Custom: {
|
|
|
|
SDOperand Tmp = TLI.LowerOperation(Result, DAG);
|
|
|
|
if (Tmp.Val) {
|
|
|
|
Result = LegalizeOp(Tmp);
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
// FALLTHROUGH if the target thinks it is legal.
|
|
|
|
}
|
|
|
|
case TargetLowering::Legal:
|
|
|
|
// Nothing to do.
|
|
|
|
break;
|
|
|
|
}
|
2005-01-07 15:47:09 +08:00
|
|
|
break;
|
|
|
|
}
|
|
|
|
case Promote:
|
2005-01-15 13:21:40 +08:00
|
|
|
// Truncate the value and store the result.
|
|
|
|
Tmp3 = PromoteOp(Node->getOperand(1));
|
|
|
|
Result = DAG.getNode(ISD::TRUNCSTORE, MVT::Other, Tmp1, Tmp3, Tmp2,
|
2005-04-28 04:10:01 +08:00
|
|
|
Node->getOperand(3),
|
2005-07-10 08:29:18 +08:00
|
|
|
DAG.getValueType(Node->getOperand(1).getValueType()));
|
2005-01-15 13:21:40 +08:00
|
|
|
break;
|
|
|
|
|
2005-01-07 15:47:09 +08:00
|
|
|
case Expand:
|
|
|
|
SDOperand Lo, Hi;
|
Check in code to scalarize arbitrarily wide packed types for some simple
vector operations (load, add, sub, mul).
This allows us to codegen:
void %foo(<4 x float> * %a) {
entry:
%tmp1 = load <4 x float> * %a;
%tmp2 = add <4 x float> %tmp1, %tmp1
store <4 x float> %tmp2, <4 x float> *%a
ret void
}
on ppc as:
_foo:
lfs f0, 12(r3)
lfs f1, 8(r3)
lfs f2, 4(r3)
lfs f3, 0(r3)
fadds f0, f0, f0
fadds f1, f1, f1
fadds f2, f2, f2
fadds f3, f3, f3
stfs f0, 12(r3)
stfs f1, 8(r3)
stfs f2, 4(r3)
stfs f3, 0(r3)
blr
llvm-svn: 24484
2005-11-23 02:16:00 +08:00
|
|
|
unsigned IncrementSize;
|
2005-01-07 15:47:09 +08:00
|
|
|
ExpandOp(Node->getOperand(1), Lo, Hi);
|
|
|
|
|
|
|
|
if (!TLI.isLittleEndian())
|
|
|
|
std::swap(Lo, Hi);
|
|
|
|
|
2005-05-11 12:51:16 +08:00
|
|
|
Lo = DAG.getNode(ISD::STORE, MVT::Other, Tmp1, Lo, Tmp2,
|
|
|
|
Node->getOperand(3));
|
Check in code to scalarize arbitrarily wide packed types for some simple
vector operations (load, add, sub, mul).
This allows us to codegen:
void %foo(<4 x float> * %a) {
entry:
%tmp1 = load <4 x float> * %a;
%tmp2 = add <4 x float> %tmp1, %tmp1
store <4 x float> %tmp2, <4 x float> *%a
ret void
}
on ppc as:
_foo:
lfs f0, 12(r3)
lfs f1, 8(r3)
lfs f2, 4(r3)
lfs f3, 0(r3)
fadds f0, f0, f0
fadds f1, f1, f1
fadds f2, f2, f2
fadds f3, f3, f3
stfs f0, 12(r3)
stfs f1, 8(r3)
stfs f2, 4(r3)
stfs f3, 0(r3)
blr
llvm-svn: 24484
2005-11-23 02:16:00 +08:00
|
|
|
// If this is a vector type, then we have to calculate the increment as
|
|
|
|
// the product of the element size in bytes, and the number of elements
|
|
|
|
// in the high half of the vector.
|
|
|
|
if (MVT::Vector == Hi.getValueType()) {
|
|
|
|
unsigned NumElems = cast<ConstantSDNode>(Hi.getOperand(2))->getValue();
|
|
|
|
MVT::ValueType EVT = cast<VTSDNode>(Hi.getOperand(3))->getVT();
|
|
|
|
IncrementSize = NumElems * MVT::getSizeInBits(EVT)/8;
|
|
|
|
} else {
|
|
|
|
IncrementSize = MVT::getSizeInBits(Hi.getValueType())/8;
|
|
|
|
}
|
2005-01-07 15:47:09 +08:00
|
|
|
Tmp2 = DAG.getNode(ISD::ADD, Tmp2.getValueType(), Tmp2,
|
|
|
|
getIntPtrConstant(IncrementSize));
|
|
|
|
assert(isTypeLegal(Tmp2.getValueType()) &&
|
|
|
|
"Pointers must be legal!");
|
2005-04-28 04:10:01 +08:00
|
|
|
//Again, claiming both parts of the store came form the same Instr
|
2005-05-11 12:51:16 +08:00
|
|
|
Hi = DAG.getNode(ISD::STORE, MVT::Other, Tmp1, Hi, Tmp2,
|
|
|
|
Node->getOperand(3));
|
2005-01-20 02:02:17 +08:00
|
|
|
Result = DAG.getNode(ISD::TokenFactor, MVT::Other, Lo, Hi);
|
|
|
|
break;
|
2005-01-07 15:47:09 +08:00
|
|
|
}
|
|
|
|
break;
|
2005-12-23 15:29:34 +08:00
|
|
|
}
|
2005-04-01 05:24:06 +08:00
|
|
|
case ISD::PCMARKER:
|
|
|
|
Tmp1 = LegalizeOp(Node->getOperand(0)); // Legalize the chain.
|
2005-04-02 13:00:07 +08:00
|
|
|
if (Tmp1 != Node->getOperand(0))
|
|
|
|
Result = DAG.getNode(ISD::PCMARKER, MVT::Other, Tmp1,Node->getOperand(1));
|
2005-04-01 05:24:06 +08:00
|
|
|
break;
|
2006-01-13 10:50:02 +08:00
|
|
|
case ISD::STACKSAVE:
|
|
|
|
Tmp1 = LegalizeOp(Node->getOperand(0)); // Legalize the chain.
|
|
|
|
if (Tmp1 != Node->getOperand(0)) {
|
|
|
|
std::vector<MVT::ValueType> VTs;
|
|
|
|
VTs.push_back(Node->getValueType(0));
|
|
|
|
VTs.push_back(MVT::Other);
|
|
|
|
std::vector<SDOperand> Ops;
|
|
|
|
Ops.push_back(Tmp1);
|
|
|
|
Result = DAG.getNode(ISD::STACKSAVE, VTs, Ops);
|
|
|
|
}
|
|
|
|
|
|
|
|
switch (TLI.getOperationAction(ISD::STACKSAVE, MVT::Other)) {
|
|
|
|
default: assert(0 && "This action is not supported yet!");
|
|
|
|
case TargetLowering::Custom: {
|
|
|
|
SDOperand Tmp = TLI.LowerOperation(Result, DAG);
|
|
|
|
if (Tmp.Val) {
|
|
|
|
Result = LegalizeOp(Tmp);
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
// FALLTHROUGH if the target thinks it is legal.
|
|
|
|
}
|
|
|
|
case TargetLowering::Legal:
|
|
|
|
// Since stacksave produce two values, make sure to remember that we
|
|
|
|
// legalized both of them.
|
|
|
|
AddLegalizedOperand(SDOperand(Node, 0), Result);
|
|
|
|
AddLegalizedOperand(SDOperand(Node, 1), Result.getValue(1));
|
|
|
|
return Result.getValue(Op.ResNo);
|
|
|
|
case TargetLowering::Expand:
|
2006-01-14 01:48:44 +08:00
|
|
|
// Expand to CopyFromReg if the target set
|
|
|
|
// StackPointerRegisterToSaveRestore.
|
|
|
|
if (unsigned SP = TLI.getStackPointerRegisterToSaveRestore()) {
|
2006-01-19 07:19:08 +08:00
|
|
|
Tmp1 = DAG.getCopyFromReg(Tmp1, SP,
|
2006-01-14 01:48:44 +08:00
|
|
|
Node->getValueType(0));
|
|
|
|
AddLegalizedOperand(SDOperand(Node, 0), Tmp1);
|
|
|
|
AddLegalizedOperand(SDOperand(Node, 1), Tmp1.getValue(1));
|
|
|
|
return Tmp1.getValue(Op.ResNo);
|
|
|
|
} else {
|
|
|
|
Tmp1 = DAG.getNode(ISD::UNDEF, Node->getValueType(0));
|
|
|
|
AddLegalizedOperand(SDOperand(Node, 0), Tmp1);
|
|
|
|
AddLegalizedOperand(SDOperand(Node, 1), Node->getOperand(0));
|
|
|
|
return Op.ResNo ? Node->getOperand(0) : Tmp1;
|
|
|
|
}
|
2006-01-13 10:50:02 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
case ISD::STACKRESTORE:
|
|
|
|
Tmp1 = LegalizeOp(Node->getOperand(0)); // Legalize the chain.
|
|
|
|
Tmp2 = LegalizeOp(Node->getOperand(1)); // Legalize the pointer.
|
|
|
|
if (Tmp1 != Node->getOperand(0) || Tmp2 != Node->getOperand(1))
|
|
|
|
Result = DAG.getNode(ISD::STACKRESTORE, MVT::Other, Tmp1, Tmp2);
|
|
|
|
|
|
|
|
switch (TLI.getOperationAction(ISD::STACKRESTORE, MVT::Other)) {
|
|
|
|
default: assert(0 && "This action is not supported yet!");
|
|
|
|
case TargetLowering::Custom: {
|
|
|
|
SDOperand Tmp = TLI.LowerOperation(Result, DAG);
|
|
|
|
if (Tmp.Val) {
|
|
|
|
Result = LegalizeOp(Tmp);
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
// FALLTHROUGH if the target thinks it is legal.
|
|
|
|
}
|
|
|
|
case TargetLowering::Legal:
|
|
|
|
break;
|
|
|
|
case TargetLowering::Expand:
|
2006-01-14 01:48:44 +08:00
|
|
|
// Expand to CopyToReg if the target set
|
|
|
|
// StackPointerRegisterToSaveRestore.
|
|
|
|
if (unsigned SP = TLI.getStackPointerRegisterToSaveRestore()) {
|
|
|
|
Result = DAG.getCopyToReg(Tmp1, SP, Tmp2);
|
|
|
|
} else {
|
|
|
|
Result = Tmp1;
|
|
|
|
}
|
2006-01-13 10:50:02 +08:00
|
|
|
break;
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
|
2005-11-12 00:47:30 +08:00
|
|
|
case ISD::READCYCLECOUNTER:
|
|
|
|
Tmp1 = LegalizeOp(Node->getOperand(0)); // Legalize the chain
|
2005-12-02 14:08:08 +08:00
|
|
|
if (Tmp1 != Node->getOperand(0)) {
|
|
|
|
std::vector<MVT::ValueType> rtypes;
|
|
|
|
std::vector<SDOperand> rvals;
|
|
|
|
rtypes.push_back(MVT::i64);
|
|
|
|
rtypes.push_back(MVT::Other);
|
|
|
|
rvals.push_back(Tmp1);
|
|
|
|
Result = DAG.getNode(ISD::READCYCLECOUNTER, rtypes, rvals);
|
|
|
|
}
|
2005-12-02 12:56:24 +08:00
|
|
|
|
|
|
|
// Since rdcc produce two values, make sure to remember that we legalized
|
|
|
|
// both of them.
|
|
|
|
AddLegalizedOperand(SDOperand(Node, 0), Result);
|
|
|
|
AddLegalizedOperand(SDOperand(Node, 1), Result.getValue(1));
|
|
|
|
return Result.getValue(Op.ResNo);
|
2005-11-21 05:32:07 +08:00
|
|
|
|
2005-12-23 15:29:34 +08:00
|
|
|
case ISD::TRUNCSTORE: {
|
2005-01-15 14:18:18 +08:00
|
|
|
Tmp1 = LegalizeOp(Node->getOperand(0)); // Legalize the chain.
|
|
|
|
Tmp3 = LegalizeOp(Node->getOperand(2)); // Legalize the pointer.
|
|
|
|
|
|
|
|
switch (getTypeAction(Node->getOperand(1).getValueType())) {
|
2005-12-24 00:12:20 +08:00
|
|
|
case Promote:
|
|
|
|
case Expand:
|
|
|
|
assert(0 && "Cannot handle illegal TRUNCSTORE yet!");
|
2005-01-15 14:18:18 +08:00
|
|
|
case Legal:
|
|
|
|
Tmp2 = LegalizeOp(Node->getOperand(1));
|
2005-09-10 08:20:18 +08:00
|
|
|
|
|
|
|
// The only promote case we handle is TRUNCSTORE:i1 X into
|
|
|
|
// -> TRUNCSTORE:i8 (and X, 1)
|
|
|
|
if (cast<VTSDNode>(Node->getOperand(4))->getVT() == MVT::i1 &&
|
|
|
|
TLI.getOperationAction(ISD::TRUNCSTORE, MVT::i1) ==
|
|
|
|
TargetLowering::Promote) {
|
|
|
|
// Promote the bool to a mask then store.
|
|
|
|
Tmp2 = DAG.getNode(ISD::AND, Tmp2.getValueType(), Tmp2,
|
|
|
|
DAG.getConstant(1, Tmp2.getValueType()));
|
|
|
|
Result = DAG.getNode(ISD::TRUNCSTORE, MVT::Other, Tmp1, Tmp2, Tmp3,
|
|
|
|
Node->getOperand(3), DAG.getValueType(MVT::i8));
|
|
|
|
|
|
|
|
} else if (Tmp1 != Node->getOperand(0) || Tmp2 != Node->getOperand(1) ||
|
|
|
|
Tmp3 != Node->getOperand(2)) {
|
2005-01-15 15:15:18 +08:00
|
|
|
Result = DAG.getNode(ISD::TRUNCSTORE, MVT::Other, Tmp1, Tmp2, Tmp3,
|
2005-07-10 08:29:18 +08:00
|
|
|
Node->getOperand(3), Node->getOperand(4));
|
2005-09-10 08:20:18 +08:00
|
|
|
}
|
2005-12-23 15:29:34 +08:00
|
|
|
|
|
|
|
MVT::ValueType StVT = cast<VTSDNode>(Result.Val->getOperand(4))->getVT();
|
|
|
|
switch (TLI.getOperationAction(Result.Val->getOpcode(), StVT)) {
|
|
|
|
default: assert(0 && "This action is not supported yet!");
|
|
|
|
case TargetLowering::Custom: {
|
|
|
|
SDOperand Tmp = TLI.LowerOperation(Result, DAG);
|
|
|
|
if (Tmp.Val) {
|
|
|
|
Result = LegalizeOp(Tmp);
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
// FALLTHROUGH if the target thinks it is legal.
|
|
|
|
}
|
|
|
|
case TargetLowering::Legal:
|
|
|
|
// Nothing to do.
|
|
|
|
break;
|
|
|
|
}
|
2005-01-15 14:18:18 +08:00
|
|
|
break;
|
|
|
|
}
|
|
|
|
break;
|
2005-12-23 15:29:34 +08:00
|
|
|
}
|
2005-01-15 06:08:15 +08:00
|
|
|
case ISD::SELECT:
|
2005-01-19 03:27:06 +08:00
|
|
|
switch (getTypeAction(Node->getOperand(0).getValueType())) {
|
|
|
|
case Expand: assert(0 && "It's impossible to expand bools");
|
|
|
|
case Legal:
|
|
|
|
Tmp1 = LegalizeOp(Node->getOperand(0)); // Legalize the condition.
|
|
|
|
break;
|
|
|
|
case Promote:
|
|
|
|
Tmp1 = PromoteOp(Node->getOperand(0)); // Promote the condition.
|
|
|
|
break;
|
|
|
|
}
|
2005-01-07 15:47:09 +08:00
|
|
|
Tmp2 = LegalizeOp(Node->getOperand(1)); // TrueVal
|
2005-01-15 06:08:15 +08:00
|
|
|
Tmp3 = LegalizeOp(Node->getOperand(2)); // FalseVal
|
2005-01-16 15:29:19 +08:00
|
|
|
|
2005-08-23 12:29:48 +08:00
|
|
|
switch (TLI.getOperationAction(ISD::SELECT, Tmp2.getValueType())) {
|
2005-01-16 15:29:19 +08:00
|
|
|
default: assert(0 && "This action is not supported yet!");
|
2005-08-11 04:51:12 +08:00
|
|
|
case TargetLowering::Expand:
|
|
|
|
if (Tmp1.getOpcode() == ISD::SETCC) {
|
|
|
|
Result = DAG.getSelectCC(Tmp1.getOperand(0), Tmp1.getOperand(1),
|
|
|
|
Tmp2, Tmp3,
|
|
|
|
cast<CondCodeSDNode>(Tmp1.getOperand(2))->get());
|
|
|
|
} else {
|
When legalizing brcond ->brcc or select -> selectcc, make sure to truncate
the old condition to a one bit value. The incoming value must have been
promoted, and the top bits are undefined. This causes us to generate:
_test:
rlwinm r2, r3, 0, 31, 31
li r3, 17
cmpwi cr0, r2, 0
bne .LBB_test_2 ;
.LBB_test_1: ;
li r3, 1
.LBB_test_2: ;
blr
instead of:
_test:
rlwinm r2, r3, 0, 31, 31
li r2, 17
cmpwi cr0, r3, 0
bne .LBB_test_2 ;
.LBB_test_1: ;
li r2, 1
.LBB_test_2: ;
or r3, r2, r2
blr
for:
int %test(bool %c) {
%retval = select bool %c, int 17, int 1
ret int %retval
}
llvm-svn: 22947
2005-08-22 02:03:09 +08:00
|
|
|
// Make sure the condition is either zero or one. It may have been
|
|
|
|
// promoted from something else.
|
|
|
|
Tmp1 = DAG.getZeroExtendInReg(Tmp1, MVT::i1);
|
2005-08-11 04:51:12 +08:00
|
|
|
Result = DAG.getSelectCC(Tmp1,
|
|
|
|
DAG.getConstant(0, Tmp1.getValueType()),
|
|
|
|
Tmp2, Tmp3, ISD::SETNE);
|
|
|
|
}
|
2005-12-19 07:54:29 +08:00
|
|
|
Result = LegalizeOp(Result); // Relegalize new nodes.
|
2005-08-11 04:51:12 +08:00
|
|
|
break;
|
2005-12-17 09:21:05 +08:00
|
|
|
case TargetLowering::Custom: {
|
|
|
|
SDOperand Tmp =
|
|
|
|
TLI.LowerOperation(DAG.getNode(ISD::SELECT, Node->getValueType(0),
|
|
|
|
Tmp1, Tmp2, Tmp3), DAG);
|
|
|
|
if (Tmp.Val) {
|
|
|
|
Result = LegalizeOp(Tmp);
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
// FALLTHROUGH if the target thinks it is legal.
|
|
|
|
}
|
2005-01-16 15:29:19 +08:00
|
|
|
case TargetLowering::Legal:
|
|
|
|
if (Tmp1 != Node->getOperand(0) || Tmp2 != Node->getOperand(1) ||
|
|
|
|
Tmp3 != Node->getOperand(2))
|
|
|
|
Result = DAG.getNode(ISD::SELECT, Node->getValueType(0),
|
|
|
|
Tmp1, Tmp2, Tmp3);
|
|
|
|
break;
|
|
|
|
case TargetLowering::Promote: {
|
|
|
|
MVT::ValueType NVT =
|
|
|
|
TLI.getTypeToPromoteTo(ISD::SELECT, Tmp2.getValueType());
|
|
|
|
unsigned ExtOp, TruncOp;
|
|
|
|
if (MVT::isInteger(Tmp2.getValueType())) {
|
2005-09-02 08:18:10 +08:00
|
|
|
ExtOp = ISD::ANY_EXTEND;
|
2005-01-16 15:29:19 +08:00
|
|
|
TruncOp = ISD::TRUNCATE;
|
|
|
|
} else {
|
|
|
|
ExtOp = ISD::FP_EXTEND;
|
|
|
|
TruncOp = ISD::FP_ROUND;
|
|
|
|
}
|
|
|
|
// Promote each of the values to the new type.
|
|
|
|
Tmp2 = DAG.getNode(ExtOp, NVT, Tmp2);
|
|
|
|
Tmp3 = DAG.getNode(ExtOp, NVT, Tmp3);
|
|
|
|
// Perform the larger operation, then round down.
|
|
|
|
Result = DAG.getNode(ISD::SELECT, NVT, Tmp1, Tmp2,Tmp3);
|
|
|
|
Result = DAG.getNode(TruncOp, Node->getValueType(0), Result);
|
2006-01-18 03:47:13 +08:00
|
|
|
Result = LegalizeOp(Result);
|
2005-01-16 15:29:19 +08:00
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
2005-01-07 15:47:09 +08:00
|
|
|
break;
|
2005-08-11 04:51:12 +08:00
|
|
|
case ISD::SELECT_CC:
|
|
|
|
Tmp3 = LegalizeOp(Node->getOperand(2)); // True
|
|
|
|
Tmp4 = LegalizeOp(Node->getOperand(3)); // False
|
|
|
|
|
2005-08-25 00:35:28 +08:00
|
|
|
if (isTypeLegal(Node->getOperand(0).getValueType())) {
|
2005-08-26 08:23:59 +08:00
|
|
|
// Everything is legal, see if we should expand this op or something.
|
|
|
|
switch (TLI.getOperationAction(ISD::SELECT_CC,
|
|
|
|
Node->getOperand(0).getValueType())) {
|
|
|
|
default: assert(0 && "This action is not supported yet!");
|
|
|
|
case TargetLowering::Custom: {
|
|
|
|
SDOperand Tmp =
|
|
|
|
TLI.LowerOperation(DAG.getNode(ISD::SELECT_CC, Node->getValueType(0),
|
|
|
|
Node->getOperand(0),
|
|
|
|
Node->getOperand(1), Tmp3, Tmp4,
|
2005-08-26 08:43:46 +08:00
|
|
|
Node->getOperand(4)), DAG);
|
2005-08-26 08:23:59 +08:00
|
|
|
if (Tmp.Val) {
|
|
|
|
Result = LegalizeOp(Tmp);
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
} // FALLTHROUGH if the target can't lower this operation after all.
|
|
|
|
case TargetLowering::Legal:
|
|
|
|
Tmp1 = LegalizeOp(Node->getOperand(0)); // LHS
|
|
|
|
Tmp2 = LegalizeOp(Node->getOperand(1)); // RHS
|
|
|
|
if (Tmp1 != Node->getOperand(0) || Tmp2 != Node->getOperand(1) ||
|
|
|
|
Tmp3 != Node->getOperand(2) || Tmp4 != Node->getOperand(3)) {
|
2005-12-22 13:23:45 +08:00
|
|
|
Result = DAG.getNode(ISD::SELECT_CC, Node->getValueType(0), Tmp1,Tmp2,
|
2005-08-26 08:23:59 +08:00
|
|
|
Tmp3, Tmp4, Node->getOperand(4));
|
|
|
|
}
|
|
|
|
break;
|
2005-08-11 04:51:12 +08:00
|
|
|
}
|
|
|
|
break;
|
|
|
|
} else {
|
|
|
|
Tmp1 = LegalizeOp(DAG.getNode(ISD::SETCC, TLI.getSetCCResultTy(),
|
|
|
|
Node->getOperand(0), // LHS
|
|
|
|
Node->getOperand(1), // RHS
|
|
|
|
Node->getOperand(4)));
|
2005-08-17 03:49:35 +08:00
|
|
|
// If we get a SETCC back from legalizing the SETCC node we just
|
|
|
|
// created, then use its LHS, RHS, and CC directly in creating a new
|
|
|
|
// node. Otherwise, select between the true and false value based on
|
|
|
|
// comparing the result of the legalized with zero.
|
|
|
|
if (Tmp1.getOpcode() == ISD::SETCC) {
|
|
|
|
Result = DAG.getNode(ISD::SELECT_CC, Tmp3.getValueType(),
|
|
|
|
Tmp1.getOperand(0), Tmp1.getOperand(1),
|
|
|
|
Tmp3, Tmp4, Tmp1.getOperand(2));
|
|
|
|
} else {
|
|
|
|
Result = DAG.getSelectCC(Tmp1,
|
|
|
|
DAG.getConstant(0, Tmp1.getValueType()),
|
|
|
|
Tmp3, Tmp4, ISD::SETNE);
|
|
|
|
}
|
2005-08-11 04:51:12 +08:00
|
|
|
}
|
|
|
|
break;
|
2005-01-07 15:47:09 +08:00
|
|
|
case ISD::SETCC:
|
|
|
|
switch (getTypeAction(Node->getOperand(0).getValueType())) {
|
|
|
|
case Legal:
|
|
|
|
Tmp1 = LegalizeOp(Node->getOperand(0)); // LHS
|
|
|
|
Tmp2 = LegalizeOp(Node->getOperand(1)); // RHS
|
|
|
|
break;
|
|
|
|
case Promote:
|
2005-01-16 06:16:26 +08:00
|
|
|
Tmp1 = PromoteOp(Node->getOperand(0)); // LHS
|
|
|
|
Tmp2 = PromoteOp(Node->getOperand(1)); // RHS
|
|
|
|
|
|
|
|
// If this is an FP compare, the operands have already been extended.
|
|
|
|
if (MVT::isInteger(Node->getOperand(0).getValueType())) {
|
|
|
|
MVT::ValueType VT = Node->getOperand(0).getValueType();
|
2005-01-16 09:11:45 +08:00
|
|
|
MVT::ValueType NVT = TLI.getTypeToTransformTo(VT);
|
2005-01-16 06:16:26 +08:00
|
|
|
|
|
|
|
// Otherwise, we have to insert explicit sign or zero extends. Note
|
|
|
|
// that we could insert sign extends for ALL conditions, but zero extend
|
|
|
|
// is cheaper on many machines (an AND instead of two shifts), so prefer
|
|
|
|
// it.
|
2005-08-10 04:20:18 +08:00
|
|
|
switch (cast<CondCodeSDNode>(Node->getOperand(2))->get()) {
|
2005-01-16 06:16:26 +08:00
|
|
|
default: assert(0 && "Unknown integer comparison!");
|
|
|
|
case ISD::SETEQ:
|
|
|
|
case ISD::SETNE:
|
|
|
|
case ISD::SETUGE:
|
|
|
|
case ISD::SETUGT:
|
|
|
|
case ISD::SETULE:
|
|
|
|
case ISD::SETULT:
|
|
|
|
// ALL of these operations will work if we either sign or zero extend
|
|
|
|
// the operands (including the unsigned comparisons!). Zero extend is
|
|
|
|
// usually a simpler/cheaper operation, so prefer it.
|
2005-04-13 10:38:47 +08:00
|
|
|
Tmp1 = DAG.getZeroExtendInReg(Tmp1, VT);
|
|
|
|
Tmp2 = DAG.getZeroExtendInReg(Tmp2, VT);
|
2005-01-16 06:16:26 +08:00
|
|
|
break;
|
|
|
|
case ISD::SETGE:
|
|
|
|
case ISD::SETGT:
|
|
|
|
case ISD::SETLT:
|
|
|
|
case ISD::SETLE:
|
2005-07-10 08:07:11 +08:00
|
|
|
Tmp1 = DAG.getNode(ISD::SIGN_EXTEND_INREG, NVT, Tmp1,
|
|
|
|
DAG.getValueType(VT));
|
|
|
|
Tmp2 = DAG.getNode(ISD::SIGN_EXTEND_INREG, NVT, Tmp2,
|
|
|
|
DAG.getValueType(VT));
|
2005-01-16 06:16:26 +08:00
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
2005-01-07 15:47:09 +08:00
|
|
|
break;
|
2005-04-22 06:36:52 +08:00
|
|
|
case Expand:
|
2005-01-07 15:47:09 +08:00
|
|
|
SDOperand LHSLo, LHSHi, RHSLo, RHSHi;
|
|
|
|
ExpandOp(Node->getOperand(0), LHSLo, LHSHi);
|
|
|
|
ExpandOp(Node->getOperand(1), RHSLo, RHSHi);
|
2005-08-10 04:20:18 +08:00
|
|
|
switch (cast<CondCodeSDNode>(Node->getOperand(2))->get()) {
|
2005-01-07 15:47:09 +08:00
|
|
|
case ISD::SETEQ:
|
|
|
|
case ISD::SETNE:
|
Emit long comparison against -1 better. Instead of this (x86):
test2:
movl 8(%esp), %eax
notl %eax
movl 4(%esp), %ecx
notl %ecx
orl %eax, %ecx
cmpl $0, %ecx
sete %al
movzbl %al, %eax
ret
or this (PPC):
_test2:
nor r2, r4, r4
nor r3, r3, r3
or r2, r2, r3
cntlzw r2, r2
srwi r3, r2, 5
blr
Emit this:
test2:
movl 8(%esp), %eax
andl 4(%esp), %eax
cmpl $-1, %eax
sete %al
movzbl %al, %eax
ret
or this:
_test2:
.LBB_test2_0: ;
and r2, r4, r3
cmpwi cr0, r2, -1
li r3, 1
li r2, 0
beq .LBB_test2_2 ;
.LBB_test2_1: ;
or r3, r2, r2
.LBB_test2_2: ;
blr
it seems like the PPC isel could do better for R32 == -1 case.
llvm-svn: 21242
2005-04-12 09:46:05 +08:00
|
|
|
if (RHSLo == RHSHi)
|
|
|
|
if (ConstantSDNode *RHSCST = dyn_cast<ConstantSDNode>(RHSLo))
|
|
|
|
if (RHSCST->isAllOnesValue()) {
|
|
|
|
// Comparison to -1.
|
|
|
|
Tmp1 = DAG.getNode(ISD::AND, LHSLo.getValueType(), LHSLo, LHSHi);
|
2005-08-23 12:29:48 +08:00
|
|
|
Tmp2 = RHSLo;
|
2005-04-22 06:36:52 +08:00
|
|
|
break;
|
Emit long comparison against -1 better. Instead of this (x86):
test2:
movl 8(%esp), %eax
notl %eax
movl 4(%esp), %ecx
notl %ecx
orl %eax, %ecx
cmpl $0, %ecx
sete %al
movzbl %al, %eax
ret
or this (PPC):
_test2:
nor r2, r4, r4
nor r3, r3, r3
or r2, r2, r3
cntlzw r2, r2
srwi r3, r2, 5
blr
Emit this:
test2:
movl 8(%esp), %eax
andl 4(%esp), %eax
cmpl $-1, %eax
sete %al
movzbl %al, %eax
ret
or this:
_test2:
.LBB_test2_0: ;
and r2, r4, r3
cmpwi cr0, r2, -1
li r3, 1
li r2, 0
beq .LBB_test2_2 ;
.LBB_test2_1: ;
or r3, r2, r2
.LBB_test2_2: ;
blr
it seems like the PPC isel could do better for R32 == -1 case.
llvm-svn: 21242
2005-04-12 09:46:05 +08:00
|
|
|
}
|
|
|
|
|
2005-01-07 15:47:09 +08:00
|
|
|
Tmp1 = DAG.getNode(ISD::XOR, LHSLo.getValueType(), LHSLo, RHSLo);
|
|
|
|
Tmp2 = DAG.getNode(ISD::XOR, LHSLo.getValueType(), LHSHi, RHSHi);
|
|
|
|
Tmp1 = DAG.getNode(ISD::OR, Tmp1.getValueType(), Tmp1, Tmp2);
|
2005-08-23 12:29:48 +08:00
|
|
|
Tmp2 = DAG.getConstant(0, Tmp1.getValueType());
|
2005-01-07 15:47:09 +08:00
|
|
|
break;
|
|
|
|
default:
|
Emit comparisons against the sign bit better. Codegen this:
bool %test1(long %X) {
%A = setlt long %X, 0
ret bool %A
}
like this:
test1:
cmpl $0, 8(%esp)
setl %al
movzbl %al, %eax
ret
instead of:
test1:
movl 8(%esp), %ecx
cmpl $0, %ecx
setl %al
movzbw %al, %ax
cmpl $0, 4(%esp)
setb %dl
movzbw %dl, %dx
cmpl $0, %ecx
cmove %dx, %ax
movzbl %al, %eax
ret
llvm-svn: 21243
2005-04-12 10:19:10 +08:00
|
|
|
// If this is a comparison of the sign bit, just look at the top part.
|
|
|
|
// X > -1, x < 0
|
|
|
|
if (ConstantSDNode *CST = dyn_cast<ConstantSDNode>(Node->getOperand(1)))
|
2005-08-10 04:20:18 +08:00
|
|
|
if ((cast<CondCodeSDNode>(Node->getOperand(2))->get() == ISD::SETLT &&
|
Emit comparisons against the sign bit better. Codegen this:
bool %test1(long %X) {
%A = setlt long %X, 0
ret bool %A
}
like this:
test1:
cmpl $0, 8(%esp)
setl %al
movzbl %al, %eax
ret
instead of:
test1:
movl 8(%esp), %ecx
cmpl $0, %ecx
setl %al
movzbw %al, %ax
cmpl $0, 4(%esp)
setb %dl
movzbw %dl, %dx
cmpl $0, %ecx
cmove %dx, %ax
movzbl %al, %eax
ret
llvm-svn: 21243
2005-04-12 10:19:10 +08:00
|
|
|
CST->getValue() == 0) || // X < 0
|
2005-08-10 04:20:18 +08:00
|
|
|
(cast<CondCodeSDNode>(Node->getOperand(2))->get() == ISD::SETGT &&
|
2005-08-23 12:29:48 +08:00
|
|
|
(CST->isAllOnesValue()))) { // X > -1
|
|
|
|
Tmp1 = LHSHi;
|
|
|
|
Tmp2 = RHSHi;
|
|
|
|
break;
|
|
|
|
}
|
Emit comparisons against the sign bit better. Codegen this:
bool %test1(long %X) {
%A = setlt long %X, 0
ret bool %A
}
like this:
test1:
cmpl $0, 8(%esp)
setl %al
movzbl %al, %eax
ret
instead of:
test1:
movl 8(%esp), %ecx
cmpl $0, %ecx
setl %al
movzbw %al, %ax
cmpl $0, 4(%esp)
setb %dl
movzbw %dl, %dx
cmpl $0, %ecx
cmove %dx, %ax
movzbl %al, %eax
ret
llvm-svn: 21243
2005-04-12 10:19:10 +08:00
|
|
|
|
2005-01-07 15:47:09 +08:00
|
|
|
// FIXME: This generated code sucks.
|
|
|
|
ISD::CondCode LowCC;
|
2005-08-10 04:20:18 +08:00
|
|
|
switch (cast<CondCodeSDNode>(Node->getOperand(2))->get()) {
|
2005-01-07 15:47:09 +08:00
|
|
|
default: assert(0 && "Unknown integer setcc!");
|
|
|
|
case ISD::SETLT:
|
|
|
|
case ISD::SETULT: LowCC = ISD::SETULT; break;
|
|
|
|
case ISD::SETGT:
|
|
|
|
case ISD::SETUGT: LowCC = ISD::SETUGT; break;
|
|
|
|
case ISD::SETLE:
|
|
|
|
case ISD::SETULE: LowCC = ISD::SETULE; break;
|
|
|
|
case ISD::SETGE:
|
|
|
|
case ISD::SETUGE: LowCC = ISD::SETUGE; break;
|
|
|
|
}
|
2005-04-22 06:36:52 +08:00
|
|
|
|
2005-01-07 15:47:09 +08:00
|
|
|
// Tmp1 = lo(op1) < lo(op2) // Always unsigned comparison
|
|
|
|
// Tmp2 = hi(op1) < hi(op2) // Signedness depends on operands
|
|
|
|
// dest = hi(op1) == hi(op2) ? Tmp1 : Tmp2;
|
|
|
|
|
|
|
|
// NOTE: on targets without efficient SELECT of bools, we can always use
|
|
|
|
// this identity: (B1 ? B2 : B3) --> (B1 & B2)|(!B1&B3)
|
2005-08-10 04:20:18 +08:00
|
|
|
Tmp1 = DAG.getSetCC(Node->getValueType(0), LHSLo, RHSLo, LowCC);
|
|
|
|
Tmp2 = DAG.getNode(ISD::SETCC, Node->getValueType(0), LHSHi, RHSHi,
|
|
|
|
Node->getOperand(2));
|
|
|
|
Result = DAG.getSetCC(Node->getValueType(0), LHSHi, RHSHi, ISD::SETEQ);
|
2005-08-23 12:29:48 +08:00
|
|
|
Result = LegalizeOp(DAG.getNode(ISD::SELECT, Tmp1.getValueType(),
|
|
|
|
Result, Tmp1, Tmp2));
|
2005-12-20 08:53:54 +08:00
|
|
|
AddLegalizedOperand(SDOperand(Node, 0), Result);
|
2005-08-23 12:29:48 +08:00
|
|
|
return Result;
|
2005-01-07 15:47:09 +08:00
|
|
|
}
|
|
|
|
}
|
2005-08-23 12:29:48 +08:00
|
|
|
|
2005-12-22 13:23:45 +08:00
|
|
|
switch(TLI.getOperationAction(ISD::SETCC,
|
|
|
|
Node->getOperand(0).getValueType())) {
|
2005-08-23 12:29:48 +08:00
|
|
|
default:
|
|
|
|
assert(0 && "Cannot handle this action for SETCC yet!");
|
|
|
|
break;
|
2005-12-01 01:12:26 +08:00
|
|
|
case TargetLowering::Promote: {
|
|
|
|
// First step, figure out the appropriate operation to use.
|
|
|
|
// Allow SETCC to not be supported for all legal data types
|
|
|
|
// Mostly this targets FP
|
|
|
|
MVT::ValueType NewInTy = Node->getOperand(0).getValueType();
|
|
|
|
MVT::ValueType OldVT = NewInTy;
|
|
|
|
|
|
|
|
// Scan for the appropriate larger type to use.
|
|
|
|
while (1) {
|
|
|
|
NewInTy = (MVT::ValueType)(NewInTy+1);
|
|
|
|
|
|
|
|
assert(MVT::isInteger(NewInTy) == MVT::isInteger(OldVT) &&
|
|
|
|
"Fell off of the edge of the integer world");
|
|
|
|
assert(MVT::isFloatingPoint(NewInTy) == MVT::isFloatingPoint(OldVT) &&
|
|
|
|
"Fell off of the edge of the floating point world");
|
|
|
|
|
|
|
|
// If the target supports SETCC of this type, use it.
|
2005-12-22 13:23:45 +08:00
|
|
|
if (TLI.isOperationLegal(ISD::SETCC, NewInTy))
|
2005-12-01 01:12:26 +08:00
|
|
|
break;
|
|
|
|
}
|
|
|
|
if (MVT::isInteger(NewInTy))
|
|
|
|
assert(0 && "Cannot promote Legal Integer SETCC yet");
|
|
|
|
else {
|
|
|
|
Tmp1 = DAG.getNode(ISD::FP_EXTEND, NewInTy, Tmp1);
|
|
|
|
Tmp2 = DAG.getNode(ISD::FP_EXTEND, NewInTy, Tmp2);
|
|
|
|
}
|
|
|
|
|
2005-08-30 04:46:51 +08:00
|
|
|
Result = DAG.getNode(ISD::SETCC, Node->getValueType(0), Tmp1, Tmp2,
|
|
|
|
Node->getOperand(2));
|
2006-01-18 03:47:13 +08:00
|
|
|
Result = LegalizeOp(Result);
|
2005-08-30 04:46:51 +08:00
|
|
|
break;
|
2005-12-01 01:12:26 +08:00
|
|
|
}
|
2005-12-22 04:21:51 +08:00
|
|
|
case TargetLowering::Custom: {
|
|
|
|
SDOperand Tmp =
|
|
|
|
TLI.LowerOperation(DAG.getNode(ISD::SETCC, Node->getValueType(0),
|
|
|
|
Tmp1, Tmp2, Node->getOperand(2)), DAG);
|
|
|
|
if (Tmp.Val) {
|
|
|
|
Result = LegalizeOp(Tmp);
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
// FALLTHROUGH if the target thinks it is legal.
|
|
|
|
}
|
2005-08-23 12:29:48 +08:00
|
|
|
case TargetLowering::Legal:
|
|
|
|
if (Tmp1 != Node->getOperand(0) || Tmp2 != Node->getOperand(1))
|
|
|
|
Result = DAG.getNode(ISD::SETCC, Node->getValueType(0), Tmp1, Tmp2,
|
|
|
|
Node->getOperand(2));
|
|
|
|
break;
|
|
|
|
case TargetLowering::Expand:
|
|
|
|
// Expand a setcc node into a select_cc of the same condition, lhs, and
|
|
|
|
// rhs that selects between const 1 (true) and const 0 (false).
|
|
|
|
MVT::ValueType VT = Node->getValueType(0);
|
|
|
|
Result = DAG.getNode(ISD::SELECT_CC, VT, Tmp1, Tmp2,
|
|
|
|
DAG.getConstant(1, VT), DAG.getConstant(0, VT),
|
|
|
|
Node->getOperand(2));
|
|
|
|
Result = LegalizeOp(Result);
|
|
|
|
break;
|
|
|
|
}
|
2005-01-07 15:47:09 +08:00
|
|
|
break;
|
|
|
|
|
2005-01-11 13:57:22 +08:00
|
|
|
case ISD::MEMSET:
|
|
|
|
case ISD::MEMCPY:
|
|
|
|
case ISD::MEMMOVE: {
|
2005-02-02 02:38:28 +08:00
|
|
|
Tmp1 = LegalizeOp(Node->getOperand(0)); // Chain
|
2005-01-29 06:29:18 +08:00
|
|
|
Tmp2 = LegalizeOp(Node->getOperand(1)); // Pointer
|
|
|
|
|
|
|
|
if (Node->getOpcode() == ISD::MEMSET) { // memset = ubyte
|
|
|
|
switch (getTypeAction(Node->getOperand(2).getValueType())) {
|
|
|
|
case Expand: assert(0 && "Cannot expand a byte!");
|
|
|
|
case Legal:
|
2005-02-02 02:38:28 +08:00
|
|
|
Tmp3 = LegalizeOp(Node->getOperand(2));
|
2005-01-29 06:29:18 +08:00
|
|
|
break;
|
|
|
|
case Promote:
|
2005-02-02 02:38:28 +08:00
|
|
|
Tmp3 = PromoteOp(Node->getOperand(2));
|
2005-01-29 06:29:18 +08:00
|
|
|
break;
|
|
|
|
}
|
|
|
|
} else {
|
2005-04-22 06:36:52 +08:00
|
|
|
Tmp3 = LegalizeOp(Node->getOperand(2)); // memcpy/move = pointer,
|
2005-01-29 06:29:18 +08:00
|
|
|
}
|
2005-02-02 11:44:41 +08:00
|
|
|
|
|
|
|
SDOperand Tmp4;
|
|
|
|
switch (getTypeAction(Node->getOperand(3).getValueType())) {
|
2005-07-13 09:42:45 +08:00
|
|
|
case Expand: {
|
|
|
|
// Length is too big, just take the lo-part of the length.
|
|
|
|
SDOperand HiPart;
|
|
|
|
ExpandOp(Node->getOperand(3), HiPart, Tmp4);
|
|
|
|
break;
|
|
|
|
}
|
2005-01-29 06:29:18 +08:00
|
|
|
case Legal:
|
|
|
|
Tmp4 = LegalizeOp(Node->getOperand(3));
|
|
|
|
break;
|
|
|
|
case Promote:
|
|
|
|
Tmp4 = PromoteOp(Node->getOperand(3));
|
2005-02-02 11:44:41 +08:00
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
|
|
|
SDOperand Tmp5;
|
|
|
|
switch (getTypeAction(Node->getOperand(4).getValueType())) { // uint
|
|
|
|
case Expand: assert(0 && "Cannot expand this yet!");
|
|
|
|
case Legal:
|
|
|
|
Tmp5 = LegalizeOp(Node->getOperand(4));
|
|
|
|
break;
|
|
|
|
case Promote:
|
2005-01-29 06:29:18 +08:00
|
|
|
Tmp5 = PromoteOp(Node->getOperand(4));
|
|
|
|
break;
|
|
|
|
}
|
2005-01-16 15:29:19 +08:00
|
|
|
|
|
|
|
switch (TLI.getOperationAction(Node->getOpcode(), MVT::Other)) {
|
|
|
|
default: assert(0 && "This action not implemented for this operation!");
|
2005-08-26 08:14:16 +08:00
|
|
|
case TargetLowering::Custom: {
|
|
|
|
SDOperand Tmp =
|
|
|
|
TLI.LowerOperation(DAG.getNode(Node->getOpcode(), MVT::Other, Tmp1,
|
|
|
|
Tmp2, Tmp3, Tmp4, Tmp5), DAG);
|
|
|
|
if (Tmp.Val) {
|
|
|
|
Result = LegalizeOp(Tmp);
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
// FALLTHROUGH if the target thinks it is legal.
|
|
|
|
}
|
2005-01-16 15:29:19 +08:00
|
|
|
case TargetLowering::Legal:
|
2005-01-11 13:57:22 +08:00
|
|
|
if (Tmp1 != Node->getOperand(0) || Tmp2 != Node->getOperand(1) ||
|
|
|
|
Tmp3 != Node->getOperand(2) || Tmp4 != Node->getOperand(3) ||
|
|
|
|
Tmp5 != Node->getOperand(4)) {
|
|
|
|
std::vector<SDOperand> Ops;
|
|
|
|
Ops.push_back(Tmp1); Ops.push_back(Tmp2); Ops.push_back(Tmp3);
|
|
|
|
Ops.push_back(Tmp4); Ops.push_back(Tmp5);
|
|
|
|
Result = DAG.getNode(Node->getOpcode(), MVT::Other, Ops);
|
|
|
|
}
|
2005-01-16 15:29:19 +08:00
|
|
|
break;
|
|
|
|
case TargetLowering::Expand: {
|
2005-01-11 13:57:22 +08:00
|
|
|
// Otherwise, the target does not support this operation. Lower the
|
|
|
|
// operation to an explicit libcall as appropriate.
|
|
|
|
MVT::ValueType IntPtr = TLI.getPointerTy();
|
|
|
|
const Type *IntPtrTy = TLI.getTargetData().getIntPtrType();
|
|
|
|
std::vector<std::pair<SDOperand, const Type*> > Args;
|
|
|
|
|
2005-01-12 22:53:45 +08:00
|
|
|
const char *FnName = 0;
|
2005-01-11 13:57:22 +08:00
|
|
|
if (Node->getOpcode() == ISD::MEMSET) {
|
|
|
|
Args.push_back(std::make_pair(Tmp2, IntPtrTy));
|
|
|
|
// Extend the ubyte argument to be an int value for the call.
|
|
|
|
Tmp3 = DAG.getNode(ISD::ZERO_EXTEND, MVT::i32, Tmp3);
|
|
|
|
Args.push_back(std::make_pair(Tmp3, Type::IntTy));
|
|
|
|
Args.push_back(std::make_pair(Tmp4, IntPtrTy));
|
|
|
|
|
|
|
|
FnName = "memset";
|
|
|
|
} else if (Node->getOpcode() == ISD::MEMCPY ||
|
|
|
|
Node->getOpcode() == ISD::MEMMOVE) {
|
|
|
|
Args.push_back(std::make_pair(Tmp2, IntPtrTy));
|
|
|
|
Args.push_back(std::make_pair(Tmp3, IntPtrTy));
|
|
|
|
Args.push_back(std::make_pair(Tmp4, IntPtrTy));
|
|
|
|
FnName = Node->getOpcode() == ISD::MEMMOVE ? "memmove" : "memcpy";
|
|
|
|
} else {
|
|
|
|
assert(0 && "Unknown op!");
|
|
|
|
}
|
2005-05-13 00:53:42 +08:00
|
|
|
|
2005-01-11 13:57:22 +08:00
|
|
|
std::pair<SDOperand,SDOperand> CallResult =
|
2005-05-14 02:50:42 +08:00
|
|
|
TLI.LowerCallTo(Tmp1, Type::VoidTy, false, CallingConv::C, false,
|
2005-01-11 13:57:22 +08:00
|
|
|
DAG.getExternalSymbol(FnName, IntPtr), Args, DAG);
|
2005-12-20 08:53:54 +08:00
|
|
|
Result = LegalizeOp(CallResult.second);
|
2005-01-16 15:29:19 +08:00
|
|
|
break;
|
|
|
|
}
|
2005-01-11 13:57:22 +08:00
|
|
|
}
|
|
|
|
break;
|
|
|
|
}
|
2005-05-10 04:23:03 +08:00
|
|
|
|
|
|
|
case ISD::READPORT:
|
|
|
|
Tmp1 = LegalizeOp(Node->getOperand(0));
|
|
|
|
Tmp2 = LegalizeOp(Node->getOperand(1));
|
2005-05-10 04:36:57 +08:00
|
|
|
|
2005-05-14 15:45:46 +08:00
|
|
|
if (Tmp1 != Node->getOperand(0) || Tmp2 != Node->getOperand(1)) {
|
|
|
|
std::vector<MVT::ValueType> VTs(Node->value_begin(), Node->value_end());
|
|
|
|
std::vector<SDOperand> Ops;
|
|
|
|
Ops.push_back(Tmp1);
|
|
|
|
Ops.push_back(Tmp2);
|
|
|
|
Result = DAG.getNode(ISD::READPORT, VTs, Ops);
|
|
|
|
} else
|
2005-05-10 04:23:03 +08:00
|
|
|
Result = SDOperand(Node, 0);
|
|
|
|
// Since these produce two values, make sure to remember that we legalized
|
|
|
|
// both of them.
|
|
|
|
AddLegalizedOperand(SDOperand(Node, 0), Result);
|
|
|
|
AddLegalizedOperand(SDOperand(Node, 1), Result.getValue(1));
|
|
|
|
return Result.getValue(Op.ResNo);
|
|
|
|
case ISD::WRITEPORT:
|
|
|
|
Tmp1 = LegalizeOp(Node->getOperand(0));
|
|
|
|
Tmp2 = LegalizeOp(Node->getOperand(1));
|
|
|
|
Tmp3 = LegalizeOp(Node->getOperand(2));
|
|
|
|
if (Tmp1 != Node->getOperand(0) || Tmp2 != Node->getOperand(1) ||
|
|
|
|
Tmp3 != Node->getOperand(2))
|
|
|
|
Result = DAG.getNode(Node->getOpcode(), MVT::Other, Tmp1, Tmp2, Tmp3);
|
|
|
|
break;
|
|
|
|
|
2005-05-10 04:36:57 +08:00
|
|
|
case ISD::READIO:
|
|
|
|
Tmp1 = LegalizeOp(Node->getOperand(0));
|
|
|
|
Tmp2 = LegalizeOp(Node->getOperand(1));
|
|
|
|
|
|
|
|
switch (TLI.getOperationAction(Node->getOpcode(), Node->getValueType(0))) {
|
|
|
|
case TargetLowering::Custom:
|
|
|
|
default: assert(0 && "This action not implemented for this operation!");
|
|
|
|
case TargetLowering::Legal:
|
2005-05-14 15:45:46 +08:00
|
|
|
if (Tmp1 != Node->getOperand(0) || Tmp2 != Node->getOperand(1)) {
|
|
|
|
std::vector<MVT::ValueType> VTs(Node->value_begin(), Node->value_end());
|
|
|
|
std::vector<SDOperand> Ops;
|
|
|
|
Ops.push_back(Tmp1);
|
|
|
|
Ops.push_back(Tmp2);
|
|
|
|
Result = DAG.getNode(ISD::READPORT, VTs, Ops);
|
|
|
|
} else
|
2005-05-10 04:36:57 +08:00
|
|
|
Result = SDOperand(Node, 0);
|
|
|
|
break;
|
|
|
|
case TargetLowering::Expand:
|
|
|
|
// Replace this with a load from memory.
|
|
|
|
Result = DAG.getLoad(Node->getValueType(0), Node->getOperand(0),
|
|
|
|
Node->getOperand(1), DAG.getSrcValue(NULL));
|
|
|
|
Result = LegalizeOp(Result);
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
|
|
|
// Since these produce two values, make sure to remember that we legalized
|
|
|
|
// both of them.
|
|
|
|
AddLegalizedOperand(SDOperand(Node, 0), Result);
|
|
|
|
AddLegalizedOperand(SDOperand(Node, 1), Result.getValue(1));
|
|
|
|
return Result.getValue(Op.ResNo);
|
|
|
|
|
|
|
|
case ISD::WRITEIO:
|
|
|
|
Tmp1 = LegalizeOp(Node->getOperand(0));
|
|
|
|
Tmp2 = LegalizeOp(Node->getOperand(1));
|
|
|
|
Tmp3 = LegalizeOp(Node->getOperand(2));
|
|
|
|
|
|
|
|
switch (TLI.getOperationAction(Node->getOpcode(),
|
|
|
|
Node->getOperand(1).getValueType())) {
|
|
|
|
case TargetLowering::Custom:
|
|
|
|
default: assert(0 && "This action not implemented for this operation!");
|
|
|
|
case TargetLowering::Legal:
|
|
|
|
if (Tmp1 != Node->getOperand(0) || Tmp2 != Node->getOperand(1) ||
|
|
|
|
Tmp3 != Node->getOperand(2))
|
|
|
|
Result = DAG.getNode(Node->getOpcode(), MVT::Other, Tmp1, Tmp2, Tmp3);
|
|
|
|
break;
|
|
|
|
case TargetLowering::Expand:
|
|
|
|
// Replace this with a store to memory.
|
|
|
|
Result = DAG.getNode(ISD::STORE, MVT::Other, Node->getOperand(0),
|
|
|
|
Node->getOperand(1), Node->getOperand(2),
|
|
|
|
DAG.getSrcValue(NULL));
|
|
|
|
Result = LegalizeOp(Result);
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
|
2005-01-21 02:52:28 +08:00
|
|
|
case ISD::ADD_PARTS:
|
2005-04-02 12:00:59 +08:00
|
|
|
case ISD::SUB_PARTS:
|
|
|
|
case ISD::SHL_PARTS:
|
|
|
|
case ISD::SRA_PARTS:
|
|
|
|
case ISD::SRL_PARTS: {
|
2005-01-21 02:52:28 +08:00
|
|
|
std::vector<SDOperand> Ops;
|
|
|
|
bool Changed = false;
|
|
|
|
for (unsigned i = 0, e = Node->getNumOperands(); i != e; ++i) {
|
|
|
|
Ops.push_back(LegalizeOp(Node->getOperand(i)));
|
|
|
|
Changed |= Ops.back() != Node->getOperand(i);
|
|
|
|
}
|
2005-05-14 15:25:05 +08:00
|
|
|
if (Changed) {
|
|
|
|
std::vector<MVT::ValueType> VTs(Node->value_begin(), Node->value_end());
|
|
|
|
Result = DAG.getNode(Node->getOpcode(), VTs, Ops);
|
|
|
|
}
|
2005-04-02 13:00:07 +08:00
|
|
|
|
2006-01-10 02:31:59 +08:00
|
|
|
switch (TLI.getOperationAction(Node->getOpcode(),
|
|
|
|
Node->getValueType(0))) {
|
|
|
|
default: assert(0 && "This action is not supported yet!");
|
|
|
|
case TargetLowering::Custom: {
|
|
|
|
SDOperand Tmp = TLI.LowerOperation(Result, DAG);
|
|
|
|
if (Tmp.Val) {
|
2006-01-11 03:43:26 +08:00
|
|
|
SDOperand Tmp2, RetVal(0,0);
|
2006-01-10 02:31:59 +08:00
|
|
|
for (unsigned i = 0, e = Node->getNumValues(); i != e; ++i) {
|
|
|
|
Tmp2 = LegalizeOp(Tmp.getValue(i));
|
|
|
|
AddLegalizedOperand(SDOperand(Node, i), Tmp2);
|
|
|
|
if (i == Op.ResNo)
|
2006-01-19 12:54:52 +08:00
|
|
|
RetVal = Tmp2;
|
2006-01-10 02:31:59 +08:00
|
|
|
}
|
2006-01-11 03:43:26 +08:00
|
|
|
assert(RetVal.Val && "Illegal result number");
|
2006-01-10 02:31:59 +08:00
|
|
|
return RetVal;
|
|
|
|
}
|
|
|
|
// FALLTHROUGH if the target thinks it is legal.
|
|
|
|
}
|
|
|
|
case TargetLowering::Legal:
|
|
|
|
// Nothing to do.
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
2005-04-02 13:00:07 +08:00
|
|
|
// Since these produce multiple values, make sure to remember that we
|
|
|
|
// legalized all of them.
|
|
|
|
for (unsigned i = 0, e = Node->getNumValues(); i != e; ++i)
|
|
|
|
AddLegalizedOperand(SDOperand(Node, i), Result.getValue(i));
|
|
|
|
return Result.getValue(Op.ResNo);
|
2005-01-21 02:52:28 +08:00
|
|
|
}
|
2005-04-02 13:00:07 +08:00
|
|
|
|
|
|
|
// Binary operators
|
2005-01-07 15:47:09 +08:00
|
|
|
case ISD::ADD:
|
|
|
|
case ISD::SUB:
|
|
|
|
case ISD::MUL:
|
2005-04-11 11:01:51 +08:00
|
|
|
case ISD::MULHS:
|
|
|
|
case ISD::MULHU:
|
2005-01-07 15:47:09 +08:00
|
|
|
case ISD::UDIV:
|
|
|
|
case ISD::SDIV:
|
|
|
|
case ISD::AND:
|
|
|
|
case ISD::OR:
|
|
|
|
case ISD::XOR:
|
2005-01-08 05:45:56 +08:00
|
|
|
case ISD::SHL:
|
|
|
|
case ISD::SRL:
|
|
|
|
case ISD::SRA:
|
2005-09-29 06:28:18 +08:00
|
|
|
case ISD::FADD:
|
|
|
|
case ISD::FSUB:
|
|
|
|
case ISD::FMUL:
|
|
|
|
case ISD::FDIV:
|
2005-01-07 15:47:09 +08:00
|
|
|
Tmp1 = LegalizeOp(Node->getOperand(0)); // LHS
|
2005-07-06 03:52:39 +08:00
|
|
|
switch (getTypeAction(Node->getOperand(1).getValueType())) {
|
|
|
|
case Expand: assert(0 && "Not possible");
|
|
|
|
case Legal:
|
|
|
|
Tmp2 = LegalizeOp(Node->getOperand(1)); // Legalize the RHS.
|
|
|
|
break;
|
|
|
|
case Promote:
|
|
|
|
Tmp2 = PromoteOp(Node->getOperand(1)); // Promote the RHS.
|
|
|
|
break;
|
|
|
|
}
|
2005-12-25 07:42:32 +08:00
|
|
|
switch (TLI.getOperationAction(Node->getOpcode(), Node->getValueType(0))) {
|
|
|
|
case TargetLowering::Custom: {
|
|
|
|
Result = DAG.getNode(Node->getOpcode(), Node->getValueType(0), Tmp1, Tmp2);
|
|
|
|
SDOperand Tmp = TLI.LowerOperation(Result, DAG);
|
|
|
|
if (Tmp.Val) {
|
|
|
|
Tmp = LegalizeOp(Tmp); // Relegalize input.
|
|
|
|
AddLegalizedOperand(Op, Tmp);
|
|
|
|
return Tmp;
|
2005-12-25 09:07:37 +08:00
|
|
|
} //else it was considered legal and we fall through
|
2005-12-25 07:42:32 +08:00
|
|
|
}
|
2005-12-25 09:07:37 +08:00
|
|
|
case TargetLowering::Legal:
|
|
|
|
if (Tmp1 != Node->getOperand(0) ||
|
|
|
|
Tmp2 != Node->getOperand(1))
|
|
|
|
Result = DAG.getNode(Node->getOpcode(), Node->getValueType(0), Tmp1,Tmp2);
|
|
|
|
break;
|
2005-12-25 07:42:32 +08:00
|
|
|
default:
|
|
|
|
assert(0 && "Operation not supported");
|
|
|
|
}
|
2005-01-07 15:47:09 +08:00
|
|
|
break;
|
2005-04-22 06:36:52 +08:00
|
|
|
|
2005-10-18 08:27:41 +08:00
|
|
|
case ISD::BUILD_PAIR: {
|
|
|
|
MVT::ValueType PairTy = Node->getValueType(0);
|
|
|
|
// TODO: handle the case where the Lo and Hi operands are not of legal type
|
|
|
|
Tmp1 = LegalizeOp(Node->getOperand(0)); // Lo
|
|
|
|
Tmp2 = LegalizeOp(Node->getOperand(1)); // Hi
|
|
|
|
switch (TLI.getOperationAction(ISD::BUILD_PAIR, PairTy)) {
|
|
|
|
case TargetLowering::Legal:
|
|
|
|
if (Tmp1 != Node->getOperand(0) || Tmp2 != Node->getOperand(1))
|
|
|
|
Result = DAG.getNode(ISD::BUILD_PAIR, PairTy, Tmp1, Tmp2);
|
|
|
|
break;
|
|
|
|
case TargetLowering::Promote:
|
|
|
|
case TargetLowering::Custom:
|
|
|
|
assert(0 && "Cannot promote/custom this yet!");
|
|
|
|
case TargetLowering::Expand:
|
|
|
|
Tmp1 = DAG.getNode(ISD::ZERO_EXTEND, PairTy, Tmp1);
|
|
|
|
Tmp2 = DAG.getNode(ISD::ANY_EXTEND, PairTy, Tmp2);
|
|
|
|
Tmp2 = DAG.getNode(ISD::SHL, PairTy, Tmp2,
|
|
|
|
DAG.getConstant(MVT::getSizeInBits(PairTy)/2,
|
|
|
|
TLI.getShiftAmountTy()));
|
|
|
|
Result = LegalizeOp(DAG.getNode(ISD::OR, PairTy, Tmp1, Tmp2));
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
2005-04-06 08:23:54 +08:00
|
|
|
case ISD::UREM:
|
|
|
|
case ISD::SREM:
|
2005-09-29 06:28:18 +08:00
|
|
|
case ISD::FREM:
|
2005-04-06 08:23:54 +08:00
|
|
|
Tmp1 = LegalizeOp(Node->getOperand(0)); // LHS
|
|
|
|
Tmp2 = LegalizeOp(Node->getOperand(1)); // RHS
|
|
|
|
switch (TLI.getOperationAction(Node->getOpcode(), Node->getValueType(0))) {
|
2005-12-25 07:42:32 +08:00
|
|
|
case TargetLowering::Custom: {
|
2006-01-16 15:59:13 +08:00
|
|
|
Result = DAG.getNode(Node->getOpcode(), Node->getValueType(0), Tmp1,Tmp2);
|
2005-12-25 07:42:32 +08:00
|
|
|
SDOperand Tmp = TLI.LowerOperation(Result, DAG);
|
|
|
|
if (Tmp.Val) {
|
2006-01-16 15:59:13 +08:00
|
|
|
Tmp = LegalizeOp(Tmp); // Relegalize input.
|
|
|
|
AddLegalizedOperand(Op, Tmp);
|
|
|
|
return Tmp;
|
2005-12-25 09:07:37 +08:00
|
|
|
} //else it was considered legal and we fall through
|
2005-12-25 07:42:32 +08:00
|
|
|
}
|
2005-12-25 09:07:37 +08:00
|
|
|
case TargetLowering::Legal:
|
|
|
|
if (Tmp1 != Node->getOperand(0) ||
|
|
|
|
Tmp2 != Node->getOperand(1))
|
|
|
|
Result = DAG.getNode(Node->getOpcode(), Node->getValueType(0), Tmp1,
|
|
|
|
Tmp2);
|
|
|
|
break;
|
|
|
|
case TargetLowering::Promote:
|
|
|
|
assert(0 && "Cannot promote handle this yet!");
|
2005-08-04 04:31:37 +08:00
|
|
|
case TargetLowering::Expand:
|
|
|
|
if (MVT::isInteger(Node->getValueType(0))) {
|
|
|
|
MVT::ValueType VT = Node->getValueType(0);
|
|
|
|
unsigned Opc = (Node->getOpcode() == ISD::UREM) ? ISD::UDIV : ISD::SDIV;
|
|
|
|
Result = DAG.getNode(Opc, VT, Tmp1, Tmp2);
|
|
|
|
Result = DAG.getNode(ISD::MUL, VT, Result, Tmp2);
|
|
|
|
Result = DAG.getNode(ISD::SUB, VT, Tmp1, Result);
|
|
|
|
} else {
|
|
|
|
// Floating point mod -> fmod libcall.
|
|
|
|
const char *FnName = Node->getValueType(0) == MVT::f32 ? "fmodf":"fmod";
|
|
|
|
SDOperand Dummy;
|
|
|
|
Result = ExpandLibCall(FnName, Node, Dummy);
|
2005-04-06 08:23:54 +08:00
|
|
|
}
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
break;
|
2005-04-02 13:00:07 +08:00
|
|
|
|
2006-01-12 05:21:00 +08:00
|
|
|
case ISD::ROTL:
|
|
|
|
case ISD::ROTR:
|
|
|
|
Tmp1 = LegalizeOp(Node->getOperand(0)); // LHS
|
|
|
|
Tmp2 = LegalizeOp(Node->getOperand(1)); // RHS
|
|
|
|
switch (TLI.getOperationAction(Node->getOpcode(), Node->getValueType(0))) {
|
|
|
|
case TargetLowering::Custom:
|
|
|
|
case TargetLowering::Promote:
|
|
|
|
case TargetLowering::Expand:
|
|
|
|
assert(0 && "Cannot handle this yet!");
|
|
|
|
case TargetLowering::Legal:
|
|
|
|
if (Tmp1 != Node->getOperand(0) ||
|
|
|
|
Tmp2 != Node->getOperand(1))
|
|
|
|
Result = DAG.getNode(Node->getOpcode(), Node->getValueType(0), Tmp1,
|
|
|
|
Tmp2);
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
|
2006-01-14 11:14:10 +08:00
|
|
|
case ISD::BSWAP:
|
|
|
|
Tmp1 = LegalizeOp(Node->getOperand(0)); // Op
|
|
|
|
switch (TLI.getOperationAction(Node->getOpcode(), Node->getValueType(0))) {
|
|
|
|
case TargetLowering::Legal:
|
|
|
|
if (Tmp1 != Node->getOperand(0))
|
|
|
|
Result = DAG.getNode(Node->getOpcode(), Node->getValueType(0), Tmp1);
|
|
|
|
break;
|
|
|
|
case TargetLowering::Promote: {
|
|
|
|
MVT::ValueType OVT = Tmp1.getValueType();
|
|
|
|
MVT::ValueType NVT = TLI.getTypeToPromoteTo(Node->getOpcode(), OVT);
|
|
|
|
unsigned DiffBits = getSizeInBits(NVT) - getSizeInBits(OVT);
|
|
|
|
|
|
|
|
Tmp1 = DAG.getNode(ISD::ZERO_EXTEND, NVT, Tmp1);
|
|
|
|
Tmp1 = DAG.getNode(ISD::BSWAP, NVT, Tmp1);
|
|
|
|
Result = DAG.getNode(ISD::SRL, NVT, Tmp1,
|
|
|
|
DAG.getConstant(DiffBits, TLI.getShiftAmountTy()));
|
2006-01-18 03:47:13 +08:00
|
|
|
Result = LegalizeOp(Result);
|
2006-01-14 11:14:10 +08:00
|
|
|
break;
|
|
|
|
}
|
|
|
|
case TargetLowering::Custom:
|
|
|
|
assert(0 && "Cannot custom legalize this yet!");
|
|
|
|
case TargetLowering::Expand: {
|
|
|
|
MVT::ValueType VT = Tmp1.getValueType();
|
|
|
|
switch (VT) {
|
|
|
|
default: assert(0 && "Unhandled Expand type in BSWAP!"); abort();
|
|
|
|
case MVT::i16:
|
|
|
|
Tmp2 = DAG.getNode(ISD::SHL, VT, Tmp1,
|
|
|
|
DAG.getConstant(8, TLI.getShiftAmountTy()));
|
|
|
|
Tmp1 = DAG.getNode(ISD::SRL, VT, Tmp1,
|
|
|
|
DAG.getConstant(8, TLI.getShiftAmountTy()));
|
|
|
|
Result = DAG.getNode(ISD::OR, VT, Tmp1, Tmp2);
|
|
|
|
break;
|
|
|
|
case MVT::i32:
|
|
|
|
Tmp4 = DAG.getNode(ISD::SHL, VT, Tmp1,
|
|
|
|
DAG.getConstant(24, TLI.getShiftAmountTy()));
|
|
|
|
Tmp3 = DAG.getNode(ISD::SHL, VT, Tmp1,
|
|
|
|
DAG.getConstant(8, TLI.getShiftAmountTy()));
|
|
|
|
Tmp2 = DAG.getNode(ISD::SRL, VT, Tmp1,
|
|
|
|
DAG.getConstant(8, TLI.getShiftAmountTy()));
|
|
|
|
Tmp1 = DAG.getNode(ISD::SRL, VT, Tmp1,
|
|
|
|
DAG.getConstant(24, TLI.getShiftAmountTy()));
|
|
|
|
Tmp3 = DAG.getNode(ISD::AND, VT, Tmp3, DAG.getConstant(0xFF0000, VT));
|
|
|
|
Tmp2 = DAG.getNode(ISD::AND, VT, Tmp2, DAG.getConstant(0xFF00, VT));
|
|
|
|
Tmp4 = DAG.getNode(ISD::OR, VT, Tmp4, Tmp3);
|
|
|
|
Tmp2 = DAG.getNode(ISD::OR, VT, Tmp2, Tmp1);
|
|
|
|
Result = DAG.getNode(ISD::OR, VT, Tmp4, Tmp2);
|
|
|
|
break;
|
2006-01-16 15:59:13 +08:00
|
|
|
case MVT::i64: {
|
|
|
|
SDOperand Tmp5, Tmp6, Tmp7, Tmp8;
|
|
|
|
Tmp8 = DAG.getNode(ISD::SHL, VT, Tmp1,
|
|
|
|
DAG.getConstant(56, TLI.getShiftAmountTy()));
|
|
|
|
Tmp7 = DAG.getNode(ISD::SHL, VT, Tmp1,
|
|
|
|
DAG.getConstant(40, TLI.getShiftAmountTy()));
|
|
|
|
Tmp6 = DAG.getNode(ISD::SHL, VT, Tmp1,
|
|
|
|
DAG.getConstant(24, TLI.getShiftAmountTy()));
|
|
|
|
Tmp5 = DAG.getNode(ISD::SHL, VT, Tmp1,
|
|
|
|
DAG.getConstant(8, TLI.getShiftAmountTy()));
|
|
|
|
Tmp4 = DAG.getNode(ISD::SRL, VT, Tmp1,
|
|
|
|
DAG.getConstant(8, TLI.getShiftAmountTy()));
|
|
|
|
Tmp3 = DAG.getNode(ISD::SRL, VT, Tmp1,
|
|
|
|
DAG.getConstant(24, TLI.getShiftAmountTy()));
|
|
|
|
Tmp2 = DAG.getNode(ISD::SRL, VT, Tmp1,
|
|
|
|
DAG.getConstant(40, TLI.getShiftAmountTy()));
|
|
|
|
Tmp1 = DAG.getNode(ISD::SRL, VT, Tmp1,
|
|
|
|
DAG.getConstant(56, TLI.getShiftAmountTy()));
|
|
|
|
Tmp7 = DAG.getNode(ISD::AND, VT, Tmp7,
|
|
|
|
DAG.getConstant(0x00FF000000000000ULL, VT));
|
|
|
|
Tmp6 = DAG.getNode(ISD::AND, VT, Tmp7,
|
|
|
|
DAG.getConstant(0x0000FF0000000000ULL, VT));
|
|
|
|
Tmp5 = DAG.getNode(ISD::AND, VT, Tmp7,
|
|
|
|
DAG.getConstant(0x000000FF00000000ULL, VT));
|
|
|
|
Tmp4 = DAG.getNode(ISD::AND, VT, Tmp7,
|
|
|
|
DAG.getConstant(0x00000000FF000000ULL, VT));
|
|
|
|
Tmp3 = DAG.getNode(ISD::AND, VT, Tmp7,
|
|
|
|
DAG.getConstant(0x0000000000FF0000ULL, VT));
|
|
|
|
Tmp2 = DAG.getNode(ISD::AND, VT, Tmp7,
|
|
|
|
DAG.getConstant(0x000000000000FF00ULL, VT));
|
|
|
|
Tmp8 = DAG.getNode(ISD::OR, VT, Tmp8, Tmp7);
|
|
|
|
Tmp6 = DAG.getNode(ISD::OR, VT, Tmp6, Tmp5);
|
|
|
|
Tmp4 = DAG.getNode(ISD::OR, VT, Tmp4, Tmp3);
|
|
|
|
Tmp2 = DAG.getNode(ISD::OR, VT, Tmp2, Tmp1);
|
|
|
|
Tmp8 = DAG.getNode(ISD::OR, VT, Tmp8, Tmp6);
|
|
|
|
Tmp4 = DAG.getNode(ISD::OR, VT, Tmp4, Tmp2);
|
|
|
|
Result = DAG.getNode(ISD::OR, VT, Tmp8, Tmp4);
|
|
|
|
break;
|
|
|
|
}
|
2006-01-14 11:14:10 +08:00
|
|
|
}
|
2006-01-18 03:47:13 +08:00
|
|
|
Result = LegalizeOp(Result);
|
2006-01-14 11:14:10 +08:00
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
|
2005-05-04 01:19:30 +08:00
|
|
|
case ISD::CTPOP:
|
|
|
|
case ISD::CTTZ:
|
|
|
|
case ISD::CTLZ:
|
|
|
|
Tmp1 = LegalizeOp(Node->getOperand(0)); // Op
|
|
|
|
switch (TLI.getOperationAction(Node->getOpcode(), Node->getValueType(0))) {
|
|
|
|
case TargetLowering::Legal:
|
|
|
|
if (Tmp1 != Node->getOperand(0))
|
|
|
|
Result = DAG.getNode(Node->getOpcode(), Node->getValueType(0), Tmp1);
|
|
|
|
break;
|
|
|
|
case TargetLowering::Promote: {
|
|
|
|
MVT::ValueType OVT = Tmp1.getValueType();
|
|
|
|
MVT::ValueType NVT = TLI.getTypeToPromoteTo(Node->getOpcode(), OVT);
|
2005-05-11 12:51:16 +08:00
|
|
|
|
|
|
|
// Zero extend the argument.
|
2005-05-04 01:19:30 +08:00
|
|
|
Tmp1 = DAG.getNode(ISD::ZERO_EXTEND, NVT, Tmp1);
|
|
|
|
// Perform the larger operation, then subtract if needed.
|
|
|
|
Tmp1 = DAG.getNode(Node->getOpcode(), Node->getValueType(0), Tmp1);
|
|
|
|
switch(Node->getOpcode())
|
|
|
|
{
|
|
|
|
case ISD::CTPOP:
|
|
|
|
Result = Tmp1;
|
|
|
|
break;
|
|
|
|
case ISD::CTTZ:
|
|
|
|
//if Tmp1 == sizeinbits(NVT) then Tmp1 = sizeinbits(Old VT)
|
2005-08-10 04:20:18 +08:00
|
|
|
Tmp2 = DAG.getSetCC(TLI.getSetCCResultTy(), Tmp1,
|
|
|
|
DAG.getConstant(getSizeInBits(NVT), NVT),
|
|
|
|
ISD::SETEQ);
|
2005-07-27 14:12:32 +08:00
|
|
|
Result = DAG.getNode(ISD::SELECT, NVT, Tmp2,
|
2005-05-04 01:19:30 +08:00
|
|
|
DAG.getConstant(getSizeInBits(OVT),NVT), Tmp1);
|
|
|
|
break;
|
|
|
|
case ISD::CTLZ:
|
|
|
|
//Tmp1 = Tmp1 - (sizeinbits(NVT) - sizeinbits(Old VT))
|
2005-07-27 14:12:32 +08:00
|
|
|
Result = DAG.getNode(ISD::SUB, NVT, Tmp1,
|
|
|
|
DAG.getConstant(getSizeInBits(NVT) -
|
2005-05-04 01:19:30 +08:00
|
|
|
getSizeInBits(OVT), NVT));
|
|
|
|
break;
|
|
|
|
}
|
2006-01-18 03:47:13 +08:00
|
|
|
Result = LegalizeOp(Result);
|
2005-05-04 01:19:30 +08:00
|
|
|
break;
|
|
|
|
}
|
|
|
|
case TargetLowering::Custom:
|
|
|
|
assert(0 && "Cannot custom handle this yet!");
|
|
|
|
case TargetLowering::Expand:
|
2005-05-05 23:55:21 +08:00
|
|
|
switch(Node->getOpcode())
|
|
|
|
{
|
|
|
|
case ISD::CTPOP: {
|
2005-05-11 13:21:31 +08:00
|
|
|
static const uint64_t mask[6] = {
|
|
|
|
0x5555555555555555ULL, 0x3333333333333333ULL,
|
|
|
|
0x0F0F0F0F0F0F0F0FULL, 0x00FF00FF00FF00FFULL,
|
|
|
|
0x0000FFFF0000FFFFULL, 0x00000000FFFFFFFFULL
|
|
|
|
};
|
2005-05-05 23:55:21 +08:00
|
|
|
MVT::ValueType VT = Tmp1.getValueType();
|
2005-05-11 13:21:31 +08:00
|
|
|
MVT::ValueType ShVT = TLI.getShiftAmountTy();
|
|
|
|
unsigned len = getSizeInBits(VT);
|
|
|
|
for (unsigned i = 0; (1U << i) <= (len / 2); ++i) {
|
2005-05-05 23:55:21 +08:00
|
|
|
//x = (x & mask[i][len/8]) + (x >> (1 << i) & mask[i][len/8])
|
2005-05-11 13:21:31 +08:00
|
|
|
Tmp2 = DAG.getConstant(mask[i], VT);
|
|
|
|
Tmp3 = DAG.getConstant(1ULL << i, ShVT);
|
2005-07-27 14:12:32 +08:00
|
|
|
Tmp1 = DAG.getNode(ISD::ADD, VT,
|
2005-05-05 23:55:21 +08:00
|
|
|
DAG.getNode(ISD::AND, VT, Tmp1, Tmp2),
|
|
|
|
DAG.getNode(ISD::AND, VT,
|
|
|
|
DAG.getNode(ISD::SRL, VT, Tmp1, Tmp3),
|
|
|
|
Tmp2));
|
|
|
|
}
|
2006-01-18 03:47:13 +08:00
|
|
|
Result = LegalizeOp(Tmp1);
|
2005-05-05 23:55:21 +08:00
|
|
|
break;
|
|
|
|
}
|
2005-05-11 16:45:08 +08:00
|
|
|
case ISD::CTLZ: {
|
|
|
|
/* for now, we do this:
|
2005-05-12 02:35:21 +08:00
|
|
|
x = x | (x >> 1);
|
|
|
|
x = x | (x >> 2);
|
|
|
|
...
|
|
|
|
x = x | (x >>16);
|
2005-07-27 14:12:32 +08:00
|
|
|
x = x | (x >>32); // for 64-bit input
|
2005-05-12 02:35:21 +08:00
|
|
|
return popcount(~x);
|
2005-07-27 14:12:32 +08:00
|
|
|
|
2005-05-12 02:35:21 +08:00
|
|
|
but see also: http://www.hackersdelight.org/HDcode/nlz.cc */
|
|
|
|
MVT::ValueType VT = Tmp1.getValueType();
|
2005-05-11 16:45:08 +08:00
|
|
|
MVT::ValueType ShVT = TLI.getShiftAmountTy();
|
|
|
|
unsigned len = getSizeInBits(VT);
|
|
|
|
for (unsigned i = 0; (1U << i) <= (len / 2); ++i) {
|
|
|
|
Tmp3 = DAG.getConstant(1ULL << i, ShVT);
|
2005-07-27 14:12:32 +08:00
|
|
|
Tmp1 = DAG.getNode(ISD::OR, VT, Tmp1,
|
2005-05-11 16:45:08 +08:00
|
|
|
DAG.getNode(ISD::SRL, VT, Tmp1, Tmp3));
|
|
|
|
}
|
|
|
|
Tmp3 = DAG.getNode(ISD::XOR, VT, Tmp1, DAG.getConstant(~0ULL, VT));
|
2005-05-12 02:35:21 +08:00
|
|
|
Result = LegalizeOp(DAG.getNode(ISD::CTPOP, VT, Tmp3));
|
2005-05-11 13:27:09 +08:00
|
|
|
break;
|
2005-05-11 16:45:08 +08:00
|
|
|
}
|
|
|
|
case ISD::CTTZ: {
|
2005-07-27 14:12:32 +08:00
|
|
|
// for now, we use: { return popcount(~x & (x - 1)); }
|
2005-05-12 07:43:56 +08:00
|
|
|
// unless the target has ctlz but not ctpop, in which case we use:
|
|
|
|
// { return 32 - nlz(~x & (x-1)); }
|
|
|
|
// see also http://www.hackersdelight.org/HDcode/ntz.cc
|
2005-05-12 02:35:21 +08:00
|
|
|
MVT::ValueType VT = Tmp1.getValueType();
|
|
|
|
Tmp2 = DAG.getConstant(~0ULL, VT);
|
2005-07-27 14:12:32 +08:00
|
|
|
Tmp3 = DAG.getNode(ISD::AND, VT,
|
2005-05-12 02:35:21 +08:00
|
|
|
DAG.getNode(ISD::XOR, VT, Tmp1, Tmp2),
|
|
|
|
DAG.getNode(ISD::SUB, VT, Tmp1,
|
|
|
|
DAG.getConstant(1, VT)));
|
2005-05-12 07:43:56 +08:00
|
|
|
// If ISD::CTLZ is legal and CTPOP isn't, then do that instead
|
2005-08-25 00:35:28 +08:00
|
|
|
if (!TLI.isOperationLegal(ISD::CTPOP, VT) &&
|
|
|
|
TLI.isOperationLegal(ISD::CTLZ, VT)) {
|
2005-07-27 14:12:32 +08:00
|
|
|
Result = LegalizeOp(DAG.getNode(ISD::SUB, VT,
|
2005-05-12 07:43:56 +08:00
|
|
|
DAG.getConstant(getSizeInBits(VT), VT),
|
|
|
|
DAG.getNode(ISD::CTLZ, VT, Tmp3)));
|
|
|
|
} else {
|
|
|
|
Result = LegalizeOp(DAG.getNode(ISD::CTPOP, VT, Tmp3));
|
|
|
|
}
|
2005-05-11 13:27:09 +08:00
|
|
|
break;
|
2005-05-11 16:45:08 +08:00
|
|
|
}
|
2005-05-05 23:55:21 +08:00
|
|
|
default:
|
|
|
|
assert(0 && "Cannot expand this yet!");
|
|
|
|
break;
|
|
|
|
}
|
2005-05-04 01:19:30 +08:00
|
|
|
break;
|
|
|
|
}
|
|
|
|
break;
|
2005-07-27 14:12:32 +08:00
|
|
|
|
2005-04-02 13:00:07 +08:00
|
|
|
// Unary operators
|
|
|
|
case ISD::FABS:
|
|
|
|
case ISD::FNEG:
|
2005-04-29 05:44:33 +08:00
|
|
|
case ISD::FSQRT:
|
|
|
|
case ISD::FSIN:
|
|
|
|
case ISD::FCOS:
|
2005-04-02 13:00:07 +08:00
|
|
|
Tmp1 = LegalizeOp(Node->getOperand(0));
|
|
|
|
switch (TLI.getOperationAction(Node->getOpcode(), Node->getValueType(0))) {
|
|
|
|
case TargetLowering::Legal:
|
|
|
|
if (Tmp1 != Node->getOperand(0))
|
|
|
|
Result = DAG.getNode(Node->getOpcode(), Node->getValueType(0), Tmp1);
|
|
|
|
break;
|
|
|
|
case TargetLowering::Promote:
|
|
|
|
case TargetLowering::Custom:
|
|
|
|
assert(0 && "Cannot promote/custom handle this yet!");
|
|
|
|
case TargetLowering::Expand:
|
2005-04-30 12:43:14 +08:00
|
|
|
switch(Node->getOpcode()) {
|
|
|
|
case ISD::FNEG: {
|
2005-04-02 13:00:07 +08:00
|
|
|
// Expand Y = FNEG(X) -> Y = SUB -0.0, X
|
|
|
|
Tmp2 = DAG.getConstantFP(-0.0, Node->getValueType(0));
|
2005-09-29 06:28:18 +08:00
|
|
|
Result = LegalizeOp(DAG.getNode(ISD::FSUB, Node->getValueType(0),
|
2005-04-02 13:00:07 +08:00
|
|
|
Tmp2, Tmp1));
|
2005-04-30 12:43:14 +08:00
|
|
|
break;
|
|
|
|
}
|
|
|
|
case ISD::FABS: {
|
2005-04-02 13:26:37 +08:00
|
|
|
// Expand Y = FABS(X) -> Y = (X >u 0.0) ? X : fneg(X).
|
|
|
|
MVT::ValueType VT = Node->getValueType(0);
|
|
|
|
Tmp2 = DAG.getConstantFP(0.0, VT);
|
2005-08-10 04:20:18 +08:00
|
|
|
Tmp2 = DAG.getSetCC(TLI.getSetCCResultTy(), Tmp1, Tmp2, ISD::SETUGT);
|
2005-04-02 13:26:37 +08:00
|
|
|
Tmp3 = DAG.getNode(ISD::FNEG, VT, Tmp1);
|
|
|
|
Result = DAG.getNode(ISD::SELECT, VT, Tmp2, Tmp1, Tmp3);
|
|
|
|
Result = LegalizeOp(Result);
|
2005-04-30 12:43:14 +08:00
|
|
|
break;
|
|
|
|
}
|
|
|
|
case ISD::FSQRT:
|
|
|
|
case ISD::FSIN:
|
|
|
|
case ISD::FCOS: {
|
|
|
|
MVT::ValueType VT = Node->getValueType(0);
|
|
|
|
const char *FnName = 0;
|
|
|
|
switch(Node->getOpcode()) {
|
|
|
|
case ISD::FSQRT: FnName = VT == MVT::f32 ? "sqrtf" : "sqrt"; break;
|
|
|
|
case ISD::FSIN: FnName = VT == MVT::f32 ? "sinf" : "sin"; break;
|
|
|
|
case ISD::FCOS: FnName = VT == MVT::f32 ? "cosf" : "cos"; break;
|
|
|
|
default: assert(0 && "Unreachable!");
|
|
|
|
}
|
2005-08-05 05:43:28 +08:00
|
|
|
SDOperand Dummy;
|
|
|
|
Result = ExpandLibCall(FnName, Node, Dummy);
|
2005-04-30 12:43:14 +08:00
|
|
|
break;
|
|
|
|
}
|
|
|
|
default:
|
2005-04-02 13:26:37 +08:00
|
|
|
assert(0 && "Unreachable!");
|
2005-04-02 13:00:07 +08:00
|
|
|
}
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
break;
|
2005-12-23 08:16:34 +08:00
|
|
|
|
|
|
|
case ISD::BIT_CONVERT:
|
|
|
|
if (!isTypeLegal(Node->getOperand(0).getValueType()))
|
|
|
|
Result = ExpandBIT_CONVERT(Node->getValueType(0), Node->getOperand(0));
|
|
|
|
else {
|
|
|
|
switch (TLI.getOperationAction(ISD::BIT_CONVERT,
|
|
|
|
Node->getOperand(0).getValueType())) {
|
|
|
|
default: assert(0 && "Unknown operation action!");
|
|
|
|
case TargetLowering::Expand:
|
|
|
|
Result = ExpandBIT_CONVERT(Node->getValueType(0), Node->getOperand(0));
|
|
|
|
break;
|
|
|
|
case TargetLowering::Legal:
|
|
|
|
Tmp1 = LegalizeOp(Node->getOperand(0));
|
|
|
|
if (Tmp1 != Node->getOperand(0))
|
|
|
|
Result = DAG.getNode(ISD::BIT_CONVERT, Node->getValueType(0), Tmp1);
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
break;
|
2005-04-02 13:00:07 +08:00
|
|
|
// Conversion operators. The source and destination have different types.
|
2005-07-29 07:31:12 +08:00
|
|
|
case ISD::SINT_TO_FP:
|
|
|
|
case ISD::UINT_TO_FP: {
|
|
|
|
bool isSigned = Node->getOpcode() == ISD::SINT_TO_FP;
|
|
|
|
switch (getTypeAction(Node->getOperand(0).getValueType())) {
|
|
|
|
case Legal:
|
2005-07-31 02:33:25 +08:00
|
|
|
switch (TLI.getOperationAction(Node->getOpcode(),
|
2005-07-29 07:31:12 +08:00
|
|
|
Node->getOperand(0).getValueType())) {
|
|
|
|
default: assert(0 && "Unknown operation action!");
|
|
|
|
case TargetLowering::Expand:
|
Added generic code expansion for [signed|unsigned] i32 to [f32|f64] casts in the
legalizer. PowerPC now uses this expansion instead of ISel version.
Example:
// signed integer to double conversion
double f1(signed x) {
return (double)x;
}
// unsigned integer to double conversion
double f2(unsigned x) {
return (double)x;
}
// signed integer to float conversion
float f3(signed x) {
return (float)x;
}
// unsigned integer to float conversion
float f4(unsigned x) {
return (float)x;
}
Byte Code:
internal fastcc double %_Z2f1i(int %x) {
entry:
%tmp.1 = cast int %x to double ; <double> [#uses=1]
ret double %tmp.1
}
internal fastcc double %_Z2f2j(uint %x) {
entry:
%tmp.1 = cast uint %x to double ; <double> [#uses=1]
ret double %tmp.1
}
internal fastcc float %_Z2f3i(int %x) {
entry:
%tmp.1 = cast int %x to float ; <float> [#uses=1]
ret float %tmp.1
}
internal fastcc float %_Z2f4j(uint %x) {
entry:
%tmp.1 = cast uint %x to float ; <float> [#uses=1]
ret float %tmp.1
}
internal fastcc double %_Z2g1i(int %x) {
entry:
%buffer = alloca [2 x uint] ; <[2 x uint]*> [#uses=3]
%tmp.0 = getelementptr [2 x uint]* %buffer, int 0, int 0 ; <uint*> [#uses=1]
store uint 1127219200, uint* %tmp.0
%tmp.2 = cast int %x to uint ; <uint> [#uses=1]
%tmp.3 = xor uint %tmp.2, 2147483648 ; <uint> [#uses=1]
%tmp.5 = getelementptr [2 x uint]* %buffer, int 0, int 1 ; <uint*> [#uses=1]
store uint %tmp.3, uint* %tmp.5
%tmp.9 = cast [2 x uint]* %buffer to double* ; <double*> [#uses=1]
%tmp.10 = load double* %tmp.9 ; <double> [#uses=1]
%tmp.13 = load double* cast (long* %signed_bias to double*) ; <double> [#uses=1]
%tmp.14 = sub double %tmp.10, %tmp.13 ; <double> [#uses=1]
ret double %tmp.14
}
internal fastcc double %_Z2g2j(uint %x) {
entry:
%buffer = alloca [2 x uint] ; <[2 x uint]*> [#uses=3]
%tmp.0 = getelementptr [2 x uint]* %buffer, int 0, int 0 ; <uint*> [#uses=1]
store uint 1127219200, uint* %tmp.0
%tmp.1 = getelementptr [2 x uint]* %buffer, int 0, int 1 ; <uint*> [#uses=1]
store uint %x, uint* %tmp.1
%tmp.4 = cast [2 x uint]* %buffer to double* ; <double*> [#uses=1]
%tmp.5 = load double* %tmp.4 ; <double> [#uses=1]
%tmp.8 = load double* cast (long* %unsigned_bias to double*) ; <double> [#uses=1]
%tmp.9 = sub double %tmp.5, %tmp.8 ; <double> [#uses=1]
ret double %tmp.9
}
internal fastcc float %_Z2g3i(int %x) {
entry:
%buffer = alloca [2 x uint] ; <[2 x uint]*> [#uses=3]
%tmp.0 = getelementptr [2 x uint]* %buffer, int 0, int 0 ; <uint*> [#uses=1]
store uint 1127219200, uint* %tmp.0
%tmp.2 = cast int %x to uint ; <uint> [#uses=1]
%tmp.3 = xor uint %tmp.2, 2147483648 ; <uint> [#uses=1]
%tmp.5 = getelementptr [2 x uint]* %buffer, int 0, int 1 ; <uint*> [#uses=1]
store uint %tmp.3, uint* %tmp.5
%tmp.9 = cast [2 x uint]* %buffer to double* ; <double*> [#uses=1]
%tmp.10 = load double* %tmp.9 ; <double> [#uses=1]
%tmp.13 = load double* cast (long* %signed_bias to double*) ; <double> [#uses=1]
%tmp.14 = sub double %tmp.10, %tmp.13 ; <double> [#uses=1]
%tmp.16 = cast double %tmp.14 to float ; <float> [#uses=1]
ret float %tmp.16
}
internal fastcc float %_Z2g4j(uint %x) {
entry:
%buffer = alloca [2 x uint] ; <[2 x uint]*> [#uses=3]
%tmp.0 = getelementptr [2 x uint]* %buffer, int 0, int 0 ; <uint*> [#uses=1]
store uint 1127219200, uint* %tmp.0
%tmp.1 = getelementptr [2 x uint]* %buffer, int 0, int 1 ; <uint*> [#uses=1]
store uint %x, uint* %tmp.1
%tmp.4 = cast [2 x uint]* %buffer to double* ; <double*> [#uses=1]
%tmp.5 = load double* %tmp.4 ; <double> [#uses=1]
%tmp.8 = load double* cast (long* %unsigned_bias to double*) ; <double> [#uses=1]
%tmp.9 = sub double %tmp.5, %tmp.8 ; <double> [#uses=1]
%tmp.11 = cast double %tmp.9 to float ; <float> [#uses=1]
ret float %tmp.11
}
PowerPC Code:
.machine ppc970
.const
.align 2
.CPIl1__Z2f1i_0: ; float 0x4330000080000000
.long 1501560836 ; float 4.5036e+15
.text
.align 2
.globl l1__Z2f1i
l1__Z2f1i:
.LBBl1__Z2f1i_0: ; entry
xoris r2, r3, 32768
stw r2, -4(r1)
lis r2, 17200
stw r2, -8(r1)
lfd f0, -8(r1)
lis r2, ha16(.CPIl1__Z2f1i_0)
lfs f1, lo16(.CPIl1__Z2f1i_0)(r2)
fsub f1, f0, f1
blr
.const
.align 2
.CPIl2__Z2f2j_0: ; float 0x4330000000000000
.long 1501560832 ; float 4.5036e+15
.text
.align 2
.globl l2__Z2f2j
l2__Z2f2j:
.LBBl2__Z2f2j_0: ; entry
stw r3, -4(r1)
lis r2, 17200
stw r2, -8(r1)
lfd f0, -8(r1)
lis r2, ha16(.CPIl2__Z2f2j_0)
lfs f1, lo16(.CPIl2__Z2f2j_0)(r2)
fsub f1, f0, f1
blr
.const
.align 2
.CPIl3__Z2f3i_0: ; float 0x4330000080000000
.long 1501560836 ; float 4.5036e+15
.text
.align 2
.globl l3__Z2f3i
l3__Z2f3i:
.LBBl3__Z2f3i_0: ; entry
xoris r2, r3, 32768
stw r2, -4(r1)
lis r2, 17200
stw r2, -8(r1)
lfd f0, -8(r1)
lis r2, ha16(.CPIl3__Z2f3i_0)
lfs f1, lo16(.CPIl3__Z2f3i_0)(r2)
fsub f0, f0, f1
frsp f1, f0
blr
.const
.align 2
.CPIl4__Z2f4j_0: ; float 0x4330000000000000
.long 1501560832 ; float 4.5036e+15
.text
.align 2
.globl l4__Z2f4j
l4__Z2f4j:
.LBBl4__Z2f4j_0: ; entry
stw r3, -4(r1)
lis r2, 17200
stw r2, -8(r1)
lfd f0, -8(r1)
lis r2, ha16(.CPIl4__Z2f4j_0)
lfs f1, lo16(.CPIl4__Z2f4j_0)(r2)
fsub f0, f0, f1
frsp f1, f0
blr
llvm-svn: 22814
2005-08-17 08:39:29 +08:00
|
|
|
Result = ExpandLegalINT_TO_FP(isSigned,
|
|
|
|
LegalizeOp(Node->getOperand(0)),
|
|
|
|
Node->getValueType(0));
|
2005-07-29 07:31:12 +08:00
|
|
|
AddLegalizedOperand(Op, Result);
|
|
|
|
return Result;
|
|
|
|
case TargetLowering::Promote:
|
|
|
|
Result = PromoteLegalINT_TO_FP(LegalizeOp(Node->getOperand(0)),
|
|
|
|
Node->getValueType(0),
|
|
|
|
isSigned);
|
|
|
|
AddLegalizedOperand(Op, Result);
|
|
|
|
return Result;
|
|
|
|
case TargetLowering::Legal:
|
|
|
|
break;
|
2005-11-30 14:43:03 +08:00
|
|
|
case TargetLowering::Custom: {
|
|
|
|
Tmp1 = LegalizeOp(Node->getOperand(0));
|
|
|
|
SDOperand Tmp =
|
|
|
|
DAG.getNode(Node->getOpcode(), Node->getValueType(0), Tmp1);
|
|
|
|
Tmp = TLI.LowerOperation(Tmp, DAG);
|
|
|
|
if (Tmp.Val) {
|
2005-12-20 08:53:54 +08:00
|
|
|
Tmp = LegalizeOp(Tmp); // Relegalize input.
|
2005-11-30 14:43:03 +08:00
|
|
|
AddLegalizedOperand(Op, Tmp);
|
|
|
|
return Tmp;
|
|
|
|
} else {
|
|
|
|
assert(0 && "Target Must Lower this");
|
|
|
|
}
|
|
|
|
}
|
2005-07-29 07:31:12 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
Tmp1 = LegalizeOp(Node->getOperand(0));
|
|
|
|
if (Tmp1 != Node->getOperand(0))
|
|
|
|
Result = DAG.getNode(Node->getOpcode(), Node->getValueType(0), Tmp1);
|
|
|
|
break;
|
|
|
|
case Expand:
|
|
|
|
Result = ExpandIntToFP(Node->getOpcode() == ISD::SINT_TO_FP,
|
|
|
|
Node->getValueType(0), Node->getOperand(0));
|
|
|
|
break;
|
|
|
|
case Promote:
|
|
|
|
if (isSigned) {
|
|
|
|
Result = PromoteOp(Node->getOperand(0));
|
|
|
|
Result = DAG.getNode(ISD::SIGN_EXTEND_INREG, Result.getValueType(),
|
|
|
|
Result, DAG.getValueType(Node->getOperand(0).getValueType()));
|
|
|
|
Result = DAG.getNode(ISD::SINT_TO_FP, Op.getValueType(), Result);
|
|
|
|
} else {
|
|
|
|
Result = PromoteOp(Node->getOperand(0));
|
|
|
|
Result = DAG.getZeroExtendInReg(Result,
|
|
|
|
Node->getOperand(0).getValueType());
|
|
|
|
Result = DAG.getNode(ISD::UINT_TO_FP, Op.getValueType(), Result);
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
case ISD::TRUNCATE:
|
|
|
|
switch (getTypeAction(Node->getOperand(0).getValueType())) {
|
|
|
|
case Legal:
|
|
|
|
Tmp1 = LegalizeOp(Node->getOperand(0));
|
|
|
|
if (Tmp1 != Node->getOperand(0))
|
|
|
|
Result = DAG.getNode(Node->getOpcode(), Node->getValueType(0), Tmp1);
|
|
|
|
break;
|
|
|
|
case Expand:
|
|
|
|
ExpandOp(Node->getOperand(0), Tmp1, Tmp2);
|
|
|
|
|
|
|
|
// Since the result is legal, we should just be able to truncate the low
|
|
|
|
// part of the source.
|
|
|
|
Result = DAG.getNode(ISD::TRUNCATE, Node->getValueType(0), Tmp1);
|
|
|
|
break;
|
|
|
|
case Promote:
|
|
|
|
Result = PromoteOp(Node->getOperand(0));
|
|
|
|
Result = DAG.getNode(ISD::TRUNCATE, Op.getValueType(), Result);
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
break;
|
2005-07-31 02:33:25 +08:00
|
|
|
|
2005-07-29 07:31:12 +08:00
|
|
|
case ISD::FP_TO_SINT:
|
|
|
|
case ISD::FP_TO_UINT:
|
|
|
|
switch (getTypeAction(Node->getOperand(0).getValueType())) {
|
|
|
|
case Legal:
|
2005-07-30 08:04:12 +08:00
|
|
|
Tmp1 = LegalizeOp(Node->getOperand(0));
|
|
|
|
|
2005-07-29 08:11:56 +08:00
|
|
|
switch (TLI.getOperationAction(Node->getOpcode(), Node->getValueType(0))){
|
|
|
|
default: assert(0 && "Unknown operation action!");
|
|
|
|
case TargetLowering::Expand:
|
Teach the legalizer how to legalize FP_TO_UINT.
Teach the legalizer to promote FP_TO_UINT to FP_TO_SINT if the wider
FP_TO_UINT is also illegal. This allows us on PPC to codegen
unsigned short foo(float a) { return a; }
as:
_foo:
.LBB_foo_0: ; entry
fctiwz f0, f1
stfd f0, -8(r1)
lwz r2, -4(r1)
rlwinm r3, r2, 0, 16, 31
blr
instead of:
_foo:
.LBB_foo_0: ; entry
fctiwz f0, f1
stfd f0, -8(r1)
lwz r2, -4(r1)
lis r3, ha16(.CPI_foo_0)
lfs f0, lo16(.CPI_foo_0)(r3)
fcmpu cr0, f1, f0
blt .LBB_foo_2 ; entry
.LBB_foo_1: ; entry
fsubs f0, f1, f0
fctiwz f0, f0
stfd f0, -16(r1)
lwz r2, -12(r1)
xoris r2, r2, 32768
.LBB_foo_2: ; entry
rlwinm r3, r2, 0, 16, 31
blr
llvm-svn: 22785
2005-08-14 09:20:53 +08:00
|
|
|
if (Node->getOpcode() == ISD::FP_TO_UINT) {
|
|
|
|
SDOperand True, False;
|
|
|
|
MVT::ValueType VT = Node->getOperand(0).getValueType();
|
|
|
|
MVT::ValueType NVT = Node->getValueType(0);
|
|
|
|
unsigned ShiftAmt = MVT::getSizeInBits(Node->getValueType(0))-1;
|
|
|
|
Tmp2 = DAG.getConstantFP((double)(1ULL << ShiftAmt), VT);
|
|
|
|
Tmp3 = DAG.getSetCC(TLI.getSetCCResultTy(),
|
|
|
|
Node->getOperand(0), Tmp2, ISD::SETLT);
|
|
|
|
True = DAG.getNode(ISD::FP_TO_SINT, NVT, Node->getOperand(0));
|
|
|
|
False = DAG.getNode(ISD::FP_TO_SINT, NVT,
|
2005-09-29 06:28:18 +08:00
|
|
|
DAG.getNode(ISD::FSUB, VT, Node->getOperand(0),
|
Teach the legalizer how to legalize FP_TO_UINT.
Teach the legalizer to promote FP_TO_UINT to FP_TO_SINT if the wider
FP_TO_UINT is also illegal. This allows us on PPC to codegen
unsigned short foo(float a) { return a; }
as:
_foo:
.LBB_foo_0: ; entry
fctiwz f0, f1
stfd f0, -8(r1)
lwz r2, -4(r1)
rlwinm r3, r2, 0, 16, 31
blr
instead of:
_foo:
.LBB_foo_0: ; entry
fctiwz f0, f1
stfd f0, -8(r1)
lwz r2, -4(r1)
lis r3, ha16(.CPI_foo_0)
lfs f0, lo16(.CPI_foo_0)(r3)
fcmpu cr0, f1, f0
blt .LBB_foo_2 ; entry
.LBB_foo_1: ; entry
fsubs f0, f1, f0
fctiwz f0, f0
stfd f0, -16(r1)
lwz r2, -12(r1)
xoris r2, r2, 32768
.LBB_foo_2: ; entry
rlwinm r3, r2, 0, 16, 31
blr
llvm-svn: 22785
2005-08-14 09:20:53 +08:00
|
|
|
Tmp2));
|
|
|
|
False = DAG.getNode(ISD::XOR, NVT, False,
|
|
|
|
DAG.getConstant(1ULL << ShiftAmt, NVT));
|
|
|
|
Result = LegalizeOp(DAG.getNode(ISD::SELECT, NVT, Tmp3, True, False));
|
2005-12-20 08:53:54 +08:00
|
|
|
AddLegalizedOperand(SDOperand(Node, 0), Result);
|
2005-08-15 02:38:32 +08:00
|
|
|
return Result;
|
Teach the legalizer how to legalize FP_TO_UINT.
Teach the legalizer to promote FP_TO_UINT to FP_TO_SINT if the wider
FP_TO_UINT is also illegal. This allows us on PPC to codegen
unsigned short foo(float a) { return a; }
as:
_foo:
.LBB_foo_0: ; entry
fctiwz f0, f1
stfd f0, -8(r1)
lwz r2, -4(r1)
rlwinm r3, r2, 0, 16, 31
blr
instead of:
_foo:
.LBB_foo_0: ; entry
fctiwz f0, f1
stfd f0, -8(r1)
lwz r2, -4(r1)
lis r3, ha16(.CPI_foo_0)
lfs f0, lo16(.CPI_foo_0)(r3)
fcmpu cr0, f1, f0
blt .LBB_foo_2 ; entry
.LBB_foo_1: ; entry
fsubs f0, f1, f0
fctiwz f0, f0
stfd f0, -16(r1)
lwz r2, -12(r1)
xoris r2, r2, 32768
.LBB_foo_2: ; entry
rlwinm r3, r2, 0, 16, 31
blr
llvm-svn: 22785
2005-08-14 09:20:53 +08:00
|
|
|
} else {
|
|
|
|
assert(0 && "Do not know how to expand FP_TO_SINT yet!");
|
|
|
|
}
|
|
|
|
break;
|
2005-07-29 08:11:56 +08:00
|
|
|
case TargetLowering::Promote:
|
2005-07-30 08:04:12 +08:00
|
|
|
Result = PromoteLegalFP_TO_INT(Tmp1, Node->getValueType(0),
|
2005-07-29 08:11:56 +08:00
|
|
|
Node->getOpcode() == ISD::FP_TO_SINT);
|
|
|
|
AddLegalizedOperand(Op, Result);
|
|
|
|
return Result;
|
2005-08-26 08:14:16 +08:00
|
|
|
case TargetLowering::Custom: {
|
|
|
|
SDOperand Tmp =
|
|
|
|
DAG.getNode(Node->getOpcode(), Node->getValueType(0), Tmp1);
|
|
|
|
Tmp = TLI.LowerOperation(Tmp, DAG);
|
|
|
|
if (Tmp.Val) {
|
2005-12-20 08:53:54 +08:00
|
|
|
Tmp = LegalizeOp(Tmp);
|
2005-08-26 08:14:16 +08:00
|
|
|
AddLegalizedOperand(Op, Tmp);
|
2005-08-30 01:30:00 +08:00
|
|
|
return Tmp;
|
2005-08-26 08:14:16 +08:00
|
|
|
} else {
|
|
|
|
// The target thinks this is legal afterall.
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
2005-07-29 08:11:56 +08:00
|
|
|
case TargetLowering::Legal:
|
|
|
|
break;
|
|
|
|
}
|
2005-07-31 02:33:25 +08:00
|
|
|
|
2005-07-29 07:31:12 +08:00
|
|
|
if (Tmp1 != Node->getOperand(0))
|
|
|
|
Result = DAG.getNode(Node->getOpcode(), Node->getValueType(0), Tmp1);
|
|
|
|
break;
|
|
|
|
case Expand:
|
|
|
|
assert(0 && "Shouldn't need to expand other operators here!");
|
|
|
|
case Promote:
|
|
|
|
Result = PromoteOp(Node->getOperand(0));
|
|
|
|
Result = DAG.getNode(Node->getOpcode(), Op.getValueType(), Result);
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
break;
|
2005-07-31 02:33:25 +08:00
|
|
|
|
2005-09-02 08:18:10 +08:00
|
|
|
case ISD::ANY_EXTEND:
|
2005-01-07 15:47:09 +08:00
|
|
|
case ISD::ZERO_EXTEND:
|
|
|
|
case ISD::SIGN_EXTEND:
|
2005-01-08 05:45:56 +08:00
|
|
|
case ISD::FP_EXTEND:
|
|
|
|
case ISD::FP_ROUND:
|
2005-01-07 15:47:09 +08:00
|
|
|
switch (getTypeAction(Node->getOperand(0).getValueType())) {
|
|
|
|
case Legal:
|
|
|
|
Tmp1 = LegalizeOp(Node->getOperand(0));
|
|
|
|
if (Tmp1 != Node->getOperand(0))
|
|
|
|
Result = DAG.getNode(Node->getOpcode(), Node->getValueType(0), Tmp1);
|
|
|
|
break;
|
2005-01-08 06:37:48 +08:00
|
|
|
case Expand:
|
2005-04-02 13:00:07 +08:00
|
|
|
assert(0 && "Shouldn't need to expand other operators here!");
|
2005-01-08 06:37:48 +08:00
|
|
|
|
2005-01-15 13:21:40 +08:00
|
|
|
case Promote:
|
|
|
|
switch (Node->getOpcode()) {
|
2005-09-02 08:18:10 +08:00
|
|
|
case ISD::ANY_EXTEND:
|
|
|
|
Result = PromoteOp(Node->getOperand(0));
|
|
|
|
Result = DAG.getNode(ISD::ANY_EXTEND, Op.getValueType(), Result);
|
|
|
|
break;
|
2005-01-16 08:38:00 +08:00
|
|
|
case ISD::ZERO_EXTEND:
|
|
|
|
Result = PromoteOp(Node->getOperand(0));
|
2005-09-02 08:18:10 +08:00
|
|
|
Result = DAG.getNode(ISD::ANY_EXTEND, Op.getValueType(), Result);
|
2005-04-13 10:38:47 +08:00
|
|
|
Result = DAG.getZeroExtendInReg(Result,
|
|
|
|
Node->getOperand(0).getValueType());
|
2005-01-15 13:21:40 +08:00
|
|
|
break;
|
|
|
|
case ISD::SIGN_EXTEND:
|
2005-01-16 08:38:00 +08:00
|
|
|
Result = PromoteOp(Node->getOperand(0));
|
2005-09-02 08:18:10 +08:00
|
|
|
Result = DAG.getNode(ISD::ANY_EXTEND, Op.getValueType(), Result);
|
2005-01-16 08:38:00 +08:00
|
|
|
Result = DAG.getNode(ISD::SIGN_EXTEND_INREG, Result.getValueType(),
|
2005-07-10 08:07:11 +08:00
|
|
|
Result,
|
|
|
|
DAG.getValueType(Node->getOperand(0).getValueType()));
|
2005-01-16 08:38:00 +08:00
|
|
|
break;
|
2005-01-15 13:21:40 +08:00
|
|
|
case ISD::FP_EXTEND:
|
2005-01-16 08:38:00 +08:00
|
|
|
Result = PromoteOp(Node->getOperand(0));
|
|
|
|
if (Result.getValueType() != Op.getValueType())
|
|
|
|
// Dynamically dead while we have only 2 FP types.
|
|
|
|
Result = DAG.getNode(ISD::FP_EXTEND, Op.getValueType(), Result);
|
|
|
|
break;
|
2005-01-15 13:21:40 +08:00
|
|
|
case ISD::FP_ROUND:
|
2005-01-16 13:06:12 +08:00
|
|
|
Result = PromoteOp(Node->getOperand(0));
|
|
|
|
Result = DAG.getNode(Node->getOpcode(), Op.getValueType(), Result);
|
|
|
|
break;
|
2005-01-15 13:21:40 +08:00
|
|
|
}
|
2005-01-07 15:47:09 +08:00
|
|
|
}
|
|
|
|
break;
|
2005-01-15 14:18:18 +08:00
|
|
|
case ISD::FP_ROUND_INREG:
|
2005-04-13 10:38:47 +08:00
|
|
|
case ISD::SIGN_EXTEND_INREG: {
|
2005-01-15 14:18:18 +08:00
|
|
|
Tmp1 = LegalizeOp(Node->getOperand(0));
|
2005-07-10 08:07:11 +08:00
|
|
|
MVT::ValueType ExtraVT = cast<VTSDNode>(Node->getOperand(1))->getVT();
|
2005-01-15 15:15:18 +08:00
|
|
|
|
|
|
|
// If this operation is not supported, convert it to a shl/shr or load/store
|
|
|
|
// pair.
|
2005-01-16 15:29:19 +08:00
|
|
|
switch (TLI.getOperationAction(Node->getOpcode(), ExtraVT)) {
|
|
|
|
default: assert(0 && "This action not supported for this op yet!");
|
|
|
|
case TargetLowering::Legal:
|
|
|
|
if (Tmp1 != Node->getOperand(0))
|
|
|
|
Result = DAG.getNode(Node->getOpcode(), Node->getValueType(0), Tmp1,
|
2005-07-10 09:55:33 +08:00
|
|
|
DAG.getValueType(ExtraVT));
|
2005-01-16 15:29:19 +08:00
|
|
|
break;
|
|
|
|
case TargetLowering::Expand:
|
2005-01-15 15:15:18 +08:00
|
|
|
// If this is an integer extend and shifts are supported, do that.
|
2005-04-13 10:38:47 +08:00
|
|
|
if (Node->getOpcode() == ISD::SIGN_EXTEND_INREG) {
|
2005-01-15 15:15:18 +08:00
|
|
|
// NOTE: we could fall back on load/store here too for targets without
|
|
|
|
// SAR. However, it is doubtful that any exist.
|
|
|
|
unsigned BitsDiff = MVT::getSizeInBits(Node->getValueType(0)) -
|
|
|
|
MVT::getSizeInBits(ExtraVT);
|
2005-01-22 08:31:52 +08:00
|
|
|
SDOperand ShiftCst = DAG.getConstant(BitsDiff, TLI.getShiftAmountTy());
|
2005-01-15 15:15:18 +08:00
|
|
|
Result = DAG.getNode(ISD::SHL, Node->getValueType(0),
|
|
|
|
Node->getOperand(0), ShiftCst);
|
|
|
|
Result = DAG.getNode(ISD::SRA, Node->getValueType(0),
|
|
|
|
Result, ShiftCst);
|
|
|
|
} else if (Node->getOpcode() == ISD::FP_ROUND_INREG) {
|
|
|
|
// The only way we can lower this is to turn it into a STORETRUNC,
|
|
|
|
// EXTLOAD pair, targetting a temporary location (a stack slot).
|
|
|
|
|
|
|
|
// NOTE: there is a choice here between constantly creating new stack
|
|
|
|
// slots and always reusing the same one. We currently always create
|
|
|
|
// new ones, as reuse may inhibit scheduling.
|
|
|
|
const Type *Ty = MVT::getTypeForValueType(ExtraVT);
|
|
|
|
unsigned TySize = (unsigned)TLI.getTargetData().getTypeSize(Ty);
|
|
|
|
unsigned Align = TLI.getTargetData().getTypeAlignment(Ty);
|
|
|
|
MachineFunction &MF = DAG.getMachineFunction();
|
2005-04-22 06:36:52 +08:00
|
|
|
int SSFI =
|
2005-01-15 15:15:18 +08:00
|
|
|
MF.getFrameInfo()->CreateStackObject((unsigned)TySize, Align);
|
|
|
|
SDOperand StackSlot = DAG.getFrameIndex(SSFI, TLI.getPointerTy());
|
|
|
|
Result = DAG.getNode(ISD::TRUNCSTORE, MVT::Other, DAG.getEntryNode(),
|
2005-05-10 04:23:03 +08:00
|
|
|
Node->getOperand(0), StackSlot,
|
2005-07-10 08:29:18 +08:00
|
|
|
DAG.getSrcValue(NULL), DAG.getValueType(ExtraVT));
|
2005-07-10 09:55:33 +08:00
|
|
|
Result = DAG.getExtLoad(ISD::EXTLOAD, Node->getValueType(0),
|
|
|
|
Result, StackSlot, DAG.getSrcValue(NULL),
|
|
|
|
ExtraVT);
|
2005-01-15 15:15:18 +08:00
|
|
|
} else {
|
|
|
|
assert(0 && "Unknown op");
|
|
|
|
}
|
|
|
|
Result = LegalizeOp(Result);
|
2005-01-16 15:29:19 +08:00
|
|
|
break;
|
2005-01-15 15:15:18 +08:00
|
|
|
}
|
2005-01-15 14:18:18 +08:00
|
|
|
break;
|
2005-01-07 15:47:09 +08:00
|
|
|
}
|
2005-01-15 15:15:18 +08:00
|
|
|
}
|
2005-01-07 15:47:09 +08:00
|
|
|
|
2005-05-13 00:53:42 +08:00
|
|
|
// Note that LegalizeOp may be reentered even from single-use nodes, which
|
|
|
|
// means that we always must cache transformed nodes.
|
|
|
|
AddLegalizedOperand(Op, Result);
|
2005-01-07 15:47:09 +08:00
|
|
|
return Result;
|
|
|
|
}
|
|
|
|
|
2005-01-16 06:16:26 +08:00
|
|
|
/// PromoteOp - Given an operation that produces a value in an invalid type,
|
|
|
|
/// promote it to compute the value into a larger type. The produced value will
|
|
|
|
/// have the correct bits for the low portion of the register, but no guarantee
|
|
|
|
/// is made about the top bits: it may be zero, sign-extended, or garbage.
|
2005-01-15 13:21:40 +08:00
|
|
|
SDOperand SelectionDAGLegalize::PromoteOp(SDOperand Op) {
|
|
|
|
MVT::ValueType VT = Op.getValueType();
|
2005-01-16 09:11:45 +08:00
|
|
|
MVT::ValueType NVT = TLI.getTypeToTransformTo(VT);
|
2005-01-15 13:21:40 +08:00
|
|
|
assert(getTypeAction(VT) == Promote &&
|
|
|
|
"Caller should expand or legalize operands that are not promotable!");
|
|
|
|
assert(NVT > VT && MVT::isInteger(NVT) == MVT::isInteger(VT) &&
|
|
|
|
"Cannot promote to smaller type!");
|
|
|
|
|
|
|
|
SDOperand Tmp1, Tmp2, Tmp3;
|
|
|
|
|
|
|
|
SDOperand Result;
|
|
|
|
SDNode *Node = Op.Val;
|
|
|
|
|
2005-09-03 04:32:45 +08:00
|
|
|
std::map<SDOperand, SDOperand>::iterator I = PromotedNodes.find(Op);
|
|
|
|
if (I != PromotedNodes.end()) return I->second;
|
2005-05-13 00:53:42 +08:00
|
|
|
|
2005-01-15 14:18:18 +08:00
|
|
|
// Promotion needs an optimization step to clean up after it, and is not
|
|
|
|
// careful to avoid operations the target does not support. Make sure that
|
|
|
|
// all generated operations are legalized in the next iteration.
|
|
|
|
NeedsAnotherIteration = true;
|
|
|
|
|
2005-01-15 13:21:40 +08:00
|
|
|
switch (Node->getOpcode()) {
|
2005-08-17 05:55:35 +08:00
|
|
|
case ISD::CopyFromReg:
|
|
|
|
assert(0 && "CopyFromReg must be legal!");
|
2005-01-15 13:21:40 +08:00
|
|
|
default:
|
|
|
|
std::cerr << "NODE: "; Node->dump(); std::cerr << "\n";
|
|
|
|
assert(0 && "Do not know how to promote this operator!");
|
|
|
|
abort();
|
2005-04-02 06:34:39 +08:00
|
|
|
case ISD::UNDEF:
|
|
|
|
Result = DAG.getNode(ISD::UNDEF, NVT);
|
|
|
|
break;
|
2005-01-15 13:21:40 +08:00
|
|
|
case ISD::Constant:
|
2005-08-31 00:56:19 +08:00
|
|
|
if (VT != MVT::i1)
|
|
|
|
Result = DAG.getNode(ISD::SIGN_EXTEND, NVT, Op);
|
|
|
|
else
|
|
|
|
Result = DAG.getNode(ISD::ZERO_EXTEND, NVT, Op);
|
2005-01-15 13:21:40 +08:00
|
|
|
assert(isa<ConstantSDNode>(Result) && "Didn't constant fold zext?");
|
|
|
|
break;
|
|
|
|
case ISD::ConstantFP:
|
|
|
|
Result = DAG.getNode(ISD::FP_EXTEND, NVT, Op);
|
|
|
|
assert(isa<ConstantFPSDNode>(Result) && "Didn't constant fold fp_extend?");
|
|
|
|
break;
|
2005-01-19 01:54:55 +08:00
|
|
|
|
2005-01-18 10:59:52 +08:00
|
|
|
case ISD::SETCC:
|
2005-08-25 00:35:28 +08:00
|
|
|
assert(isTypeLegal(TLI.getSetCCResultTy()) && "SetCC type is not legal??");
|
2005-08-10 04:20:18 +08:00
|
|
|
Result = DAG.getNode(ISD::SETCC, TLI.getSetCCResultTy(),Node->getOperand(0),
|
|
|
|
Node->getOperand(1), Node->getOperand(2));
|
2005-01-18 10:59:52 +08:00
|
|
|
Result = LegalizeOp(Result);
|
|
|
|
break;
|
2005-01-15 13:21:40 +08:00
|
|
|
|
|
|
|
case ISD::TRUNCATE:
|
|
|
|
switch (getTypeAction(Node->getOperand(0).getValueType())) {
|
|
|
|
case Legal:
|
|
|
|
Result = LegalizeOp(Node->getOperand(0));
|
|
|
|
assert(Result.getValueType() >= NVT &&
|
|
|
|
"This truncation doesn't make sense!");
|
|
|
|
if (Result.getValueType() > NVT) // Truncate to NVT instead of VT
|
|
|
|
Result = DAG.getNode(ISD::TRUNCATE, NVT, Result);
|
|
|
|
break;
|
2005-01-29 06:52:50 +08:00
|
|
|
case Promote:
|
|
|
|
// The truncation is not required, because we don't guarantee anything
|
|
|
|
// about high bits anyway.
|
|
|
|
Result = PromoteOp(Node->getOperand(0));
|
|
|
|
break;
|
2005-01-15 13:21:40 +08:00
|
|
|
case Expand:
|
2005-04-04 08:57:08 +08:00
|
|
|
ExpandOp(Node->getOperand(0), Tmp1, Tmp2);
|
|
|
|
// Truncate the low part of the expanded value to the result type
|
2005-08-02 02:16:37 +08:00
|
|
|
Result = DAG.getNode(ISD::TRUNCATE, NVT, Tmp1);
|
2005-01-15 13:21:40 +08:00
|
|
|
}
|
|
|
|
break;
|
2005-01-16 06:16:26 +08:00
|
|
|
case ISD::SIGN_EXTEND:
|
|
|
|
case ISD::ZERO_EXTEND:
|
2005-09-02 08:18:10 +08:00
|
|
|
case ISD::ANY_EXTEND:
|
2005-01-16 06:16:26 +08:00
|
|
|
switch (getTypeAction(Node->getOperand(0).getValueType())) {
|
|
|
|
case Expand: assert(0 && "BUG: Smaller reg should have been promoted!");
|
|
|
|
case Legal:
|
|
|
|
// Input is legal? Just do extend all the way to the larger type.
|
|
|
|
Result = LegalizeOp(Node->getOperand(0));
|
|
|
|
Result = DAG.getNode(Node->getOpcode(), NVT, Result);
|
|
|
|
break;
|
|
|
|
case Promote:
|
|
|
|
// Promote the reg if it's smaller.
|
|
|
|
Result = PromoteOp(Node->getOperand(0));
|
|
|
|
// The high bits are not guaranteed to be anything. Insert an extend.
|
|
|
|
if (Node->getOpcode() == ISD::SIGN_EXTEND)
|
2005-02-05 02:39:19 +08:00
|
|
|
Result = DAG.getNode(ISD::SIGN_EXTEND_INREG, NVT, Result,
|
2005-07-10 08:07:11 +08:00
|
|
|
DAG.getValueType(Node->getOperand(0).getValueType()));
|
2005-09-02 08:18:10 +08:00
|
|
|
else if (Node->getOpcode() == ISD::ZERO_EXTEND)
|
2005-04-13 10:38:47 +08:00
|
|
|
Result = DAG.getZeroExtendInReg(Result,
|
|
|
|
Node->getOperand(0).getValueType());
|
2005-01-16 06:16:26 +08:00
|
|
|
break;
|
|
|
|
}
|
|
|
|
break;
|
2005-12-23 08:16:34 +08:00
|
|
|
case ISD::BIT_CONVERT:
|
|
|
|
Result = ExpandBIT_CONVERT(Node->getValueType(0), Node->getOperand(0));
|
|
|
|
Result = PromoteOp(Result);
|
|
|
|
break;
|
|
|
|
|
2005-01-16 06:16:26 +08:00
|
|
|
case ISD::FP_EXTEND:
|
|
|
|
assert(0 && "Case not implemented. Dynamically dead with 2 FP types!");
|
|
|
|
case ISD::FP_ROUND:
|
|
|
|
switch (getTypeAction(Node->getOperand(0).getValueType())) {
|
|
|
|
case Expand: assert(0 && "BUG: Cannot expand FP regs!");
|
|
|
|
case Promote: assert(0 && "Unreachable with 2 FP types!");
|
|
|
|
case Legal:
|
|
|
|
// Input is legal? Do an FP_ROUND_INREG.
|
|
|
|
Result = LegalizeOp(Node->getOperand(0));
|
2005-07-10 08:07:11 +08:00
|
|
|
Result = DAG.getNode(ISD::FP_ROUND_INREG, NVT, Result,
|
|
|
|
DAG.getValueType(VT));
|
2005-01-16 06:16:26 +08:00
|
|
|
break;
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
|
|
|
|
case ISD::SINT_TO_FP:
|
|
|
|
case ISD::UINT_TO_FP:
|
|
|
|
switch (getTypeAction(Node->getOperand(0).getValueType())) {
|
|
|
|
case Legal:
|
|
|
|
Result = LegalizeOp(Node->getOperand(0));
|
2005-01-21 14:05:23 +08:00
|
|
|
// No extra round required here.
|
|
|
|
Result = DAG.getNode(Node->getOpcode(), NVT, Result);
|
2005-01-16 06:16:26 +08:00
|
|
|
break;
|
|
|
|
|
|
|
|
case Promote:
|
|
|
|
Result = PromoteOp(Node->getOperand(0));
|
|
|
|
if (Node->getOpcode() == ISD::SINT_TO_FP)
|
|
|
|
Result = DAG.getNode(ISD::SIGN_EXTEND_INREG, Result.getValueType(),
|
2005-07-10 08:07:11 +08:00
|
|
|
Result,
|
|
|
|
DAG.getValueType(Node->getOperand(0).getValueType()));
|
2005-01-16 06:16:26 +08:00
|
|
|
else
|
2005-04-13 10:38:47 +08:00
|
|
|
Result = DAG.getZeroExtendInReg(Result,
|
|
|
|
Node->getOperand(0).getValueType());
|
2005-01-21 14:05:23 +08:00
|
|
|
// No extra round required here.
|
|
|
|
Result = DAG.getNode(Node->getOpcode(), NVT, Result);
|
2005-01-16 06:16:26 +08:00
|
|
|
break;
|
|
|
|
case Expand:
|
2005-01-21 14:05:23 +08:00
|
|
|
Result = ExpandIntToFP(Node->getOpcode() == ISD::SINT_TO_FP, NVT,
|
|
|
|
Node->getOperand(0));
|
|
|
|
// Round if we cannot tolerate excess precision.
|
|
|
|
if (NoExcessFPPrecision)
|
2005-07-10 08:07:11 +08:00
|
|
|
Result = DAG.getNode(ISD::FP_ROUND_INREG, NVT, Result,
|
|
|
|
DAG.getValueType(VT));
|
2005-01-21 14:05:23 +08:00
|
|
|
break;
|
2005-01-16 06:16:26 +08:00
|
|
|
}
|
|
|
|
break;
|
|
|
|
|
2005-12-10 01:32:47 +08:00
|
|
|
case ISD::SIGN_EXTEND_INREG:
|
|
|
|
Result = PromoteOp(Node->getOperand(0));
|
|
|
|
Result = DAG.getNode(ISD::SIGN_EXTEND_INREG, NVT, Result,
|
|
|
|
Node->getOperand(1));
|
|
|
|
break;
|
2005-01-16 06:16:26 +08:00
|
|
|
case ISD::FP_TO_SINT:
|
|
|
|
case ISD::FP_TO_UINT:
|
|
|
|
switch (getTypeAction(Node->getOperand(0).getValueType())) {
|
|
|
|
case Legal:
|
|
|
|
Tmp1 = LegalizeOp(Node->getOperand(0));
|
|
|
|
break;
|
|
|
|
case Promote:
|
|
|
|
// The input result is prerounded, so we don't have to do anything
|
|
|
|
// special.
|
|
|
|
Tmp1 = PromoteOp(Node->getOperand(0));
|
|
|
|
break;
|
|
|
|
case Expand:
|
|
|
|
assert(0 && "not implemented");
|
|
|
|
}
|
Teach the legalizer how to legalize FP_TO_UINT.
Teach the legalizer to promote FP_TO_UINT to FP_TO_SINT if the wider
FP_TO_UINT is also illegal. This allows us on PPC to codegen
unsigned short foo(float a) { return a; }
as:
_foo:
.LBB_foo_0: ; entry
fctiwz f0, f1
stfd f0, -8(r1)
lwz r2, -4(r1)
rlwinm r3, r2, 0, 16, 31
blr
instead of:
_foo:
.LBB_foo_0: ; entry
fctiwz f0, f1
stfd f0, -8(r1)
lwz r2, -4(r1)
lis r3, ha16(.CPI_foo_0)
lfs f0, lo16(.CPI_foo_0)(r3)
fcmpu cr0, f1, f0
blt .LBB_foo_2 ; entry
.LBB_foo_1: ; entry
fsubs f0, f1, f0
fctiwz f0, f0
stfd f0, -16(r1)
lwz r2, -12(r1)
xoris r2, r2, 32768
.LBB_foo_2: ; entry
rlwinm r3, r2, 0, 16, 31
blr
llvm-svn: 22785
2005-08-14 09:20:53 +08:00
|
|
|
// If we're promoting a UINT to a larger size, check to see if the new node
|
|
|
|
// will be legal. If it isn't, check to see if FP_TO_SINT is legal, since
|
|
|
|
// we can use that instead. This allows us to generate better code for
|
|
|
|
// FP_TO_UINT for small destination sizes on targets where FP_TO_UINT is not
|
|
|
|
// legal, such as PowerPC.
|
|
|
|
if (Node->getOpcode() == ISD::FP_TO_UINT &&
|
2005-08-25 00:35:28 +08:00
|
|
|
!TLI.isOperationLegal(ISD::FP_TO_UINT, NVT) &&
|
2005-10-26 07:47:25 +08:00
|
|
|
(TLI.isOperationLegal(ISD::FP_TO_SINT, NVT) ||
|
|
|
|
TLI.getOperationAction(ISD::FP_TO_SINT, NVT)==TargetLowering::Custom)){
|
Teach the legalizer how to legalize FP_TO_UINT.
Teach the legalizer to promote FP_TO_UINT to FP_TO_SINT if the wider
FP_TO_UINT is also illegal. This allows us on PPC to codegen
unsigned short foo(float a) { return a; }
as:
_foo:
.LBB_foo_0: ; entry
fctiwz f0, f1
stfd f0, -8(r1)
lwz r2, -4(r1)
rlwinm r3, r2, 0, 16, 31
blr
instead of:
_foo:
.LBB_foo_0: ; entry
fctiwz f0, f1
stfd f0, -8(r1)
lwz r2, -4(r1)
lis r3, ha16(.CPI_foo_0)
lfs f0, lo16(.CPI_foo_0)(r3)
fcmpu cr0, f1, f0
blt .LBB_foo_2 ; entry
.LBB_foo_1: ; entry
fsubs f0, f1, f0
fctiwz f0, f0
stfd f0, -16(r1)
lwz r2, -12(r1)
xoris r2, r2, 32768
.LBB_foo_2: ; entry
rlwinm r3, r2, 0, 16, 31
blr
llvm-svn: 22785
2005-08-14 09:20:53 +08:00
|
|
|
Result = DAG.getNode(ISD::FP_TO_SINT, NVT, Tmp1);
|
|
|
|
} else {
|
|
|
|
Result = DAG.getNode(Node->getOpcode(), NVT, Tmp1);
|
|
|
|
}
|
2005-01-16 06:16:26 +08:00
|
|
|
break;
|
|
|
|
|
2005-04-02 13:00:07 +08:00
|
|
|
case ISD::FABS:
|
|
|
|
case ISD::FNEG:
|
|
|
|
Tmp1 = PromoteOp(Node->getOperand(0));
|
|
|
|
assert(Tmp1.getValueType() == NVT);
|
|
|
|
Result = DAG.getNode(Node->getOpcode(), NVT, Tmp1);
|
|
|
|
// NOTE: we do not have to do any extra rounding here for
|
|
|
|
// NoExcessFPPrecision, because we know the input will have the appropriate
|
|
|
|
// precision, and these operations don't modify precision at all.
|
|
|
|
break;
|
|
|
|
|
2005-04-29 05:44:33 +08:00
|
|
|
case ISD::FSQRT:
|
|
|
|
case ISD::FSIN:
|
|
|
|
case ISD::FCOS:
|
|
|
|
Tmp1 = PromoteOp(Node->getOperand(0));
|
|
|
|
assert(Tmp1.getValueType() == NVT);
|
|
|
|
Result = DAG.getNode(Node->getOpcode(), NVT, Tmp1);
|
|
|
|
if(NoExcessFPPrecision)
|
2005-07-10 08:07:11 +08:00
|
|
|
Result = DAG.getNode(ISD::FP_ROUND_INREG, NVT, Result,
|
|
|
|
DAG.getValueType(VT));
|
2005-04-29 05:44:33 +08:00
|
|
|
break;
|
|
|
|
|
2005-01-15 13:21:40 +08:00
|
|
|
case ISD::AND:
|
|
|
|
case ISD::OR:
|
|
|
|
case ISD::XOR:
|
2005-01-15 14:18:18 +08:00
|
|
|
case ISD::ADD:
|
2005-01-16 06:16:26 +08:00
|
|
|
case ISD::SUB:
|
2005-01-15 14:18:18 +08:00
|
|
|
case ISD::MUL:
|
|
|
|
// The input may have strange things in the top bits of the registers, but
|
2005-09-29 06:28:18 +08:00
|
|
|
// these operations don't care. They may have weird bits going out, but
|
2005-01-15 14:18:18 +08:00
|
|
|
// that too is okay if they are integer operations.
|
|
|
|
Tmp1 = PromoteOp(Node->getOperand(0));
|
|
|
|
Tmp2 = PromoteOp(Node->getOperand(1));
|
|
|
|
assert(Tmp1.getValueType() == NVT && Tmp2.getValueType() == NVT);
|
|
|
|
Result = DAG.getNode(Node->getOpcode(), NVT, Tmp1, Tmp2);
|
2005-09-29 06:28:18 +08:00
|
|
|
break;
|
|
|
|
case ISD::FADD:
|
|
|
|
case ISD::FSUB:
|
|
|
|
case ISD::FMUL:
|
|
|
|
// The input may have strange things in the top bits of the registers, but
|
|
|
|
// these operations don't care.
|
|
|
|
Tmp1 = PromoteOp(Node->getOperand(0));
|
|
|
|
Tmp2 = PromoteOp(Node->getOperand(1));
|
|
|
|
assert(Tmp1.getValueType() == NVT && Tmp2.getValueType() == NVT);
|
|
|
|
Result = DAG.getNode(Node->getOpcode(), NVT, Tmp1, Tmp2);
|
|
|
|
|
|
|
|
// Floating point operations will give excess precision that we may not be
|
|
|
|
// able to tolerate. If we DO allow excess precision, just leave it,
|
|
|
|
// otherwise excise it.
|
2005-01-16 06:16:26 +08:00
|
|
|
// FIXME: Why would we need to round FP ops more than integer ones?
|
|
|
|
// Is Round(Add(Add(A,B),C)) != Round(Add(Round(Add(A,B)), C))
|
2005-09-29 06:28:18 +08:00
|
|
|
if (NoExcessFPPrecision)
|
2005-07-10 08:07:11 +08:00
|
|
|
Result = DAG.getNode(ISD::FP_ROUND_INREG, NVT, Result,
|
|
|
|
DAG.getValueType(VT));
|
2005-01-15 14:18:18 +08:00
|
|
|
break;
|
|
|
|
|
2005-01-16 06:16:26 +08:00
|
|
|
case ISD::SDIV:
|
|
|
|
case ISD::SREM:
|
|
|
|
// These operators require that their input be sign extended.
|
|
|
|
Tmp1 = PromoteOp(Node->getOperand(0));
|
|
|
|
Tmp2 = PromoteOp(Node->getOperand(1));
|
|
|
|
if (MVT::isInteger(NVT)) {
|
2005-07-10 08:07:11 +08:00
|
|
|
Tmp1 = DAG.getNode(ISD::SIGN_EXTEND_INREG, NVT, Tmp1,
|
|
|
|
DAG.getValueType(VT));
|
|
|
|
Tmp2 = DAG.getNode(ISD::SIGN_EXTEND_INREG, NVT, Tmp2,
|
|
|
|
DAG.getValueType(VT));
|
2005-01-16 06:16:26 +08:00
|
|
|
}
|
|
|
|
Result = DAG.getNode(Node->getOpcode(), NVT, Tmp1, Tmp2);
|
|
|
|
|
|
|
|
// Perform FP_ROUND: this is probably overly pessimistic.
|
|
|
|
if (MVT::isFloatingPoint(NVT) && NoExcessFPPrecision)
|
2005-07-10 08:07:11 +08:00
|
|
|
Result = DAG.getNode(ISD::FP_ROUND_INREG, NVT, Result,
|
|
|
|
DAG.getValueType(VT));
|
2005-01-16 06:16:26 +08:00
|
|
|
break;
|
2005-09-29 06:28:18 +08:00
|
|
|
case ISD::FDIV:
|
|
|
|
case ISD::FREM:
|
|
|
|
// These operators require that their input be fp extended.
|
|
|
|
Tmp1 = PromoteOp(Node->getOperand(0));
|
|
|
|
Tmp2 = PromoteOp(Node->getOperand(1));
|
|
|
|
Result = DAG.getNode(Node->getOpcode(), NVT, Tmp1, Tmp2);
|
|
|
|
|
|
|
|
// Perform FP_ROUND: this is probably overly pessimistic.
|
|
|
|
if (NoExcessFPPrecision)
|
|
|
|
Result = DAG.getNode(ISD::FP_ROUND_INREG, NVT, Result,
|
|
|
|
DAG.getValueType(VT));
|
|
|
|
break;
|
2005-01-16 06:16:26 +08:00
|
|
|
|
|
|
|
case ISD::UDIV:
|
|
|
|
case ISD::UREM:
|
|
|
|
// These operators require that their input be zero extended.
|
|
|
|
Tmp1 = PromoteOp(Node->getOperand(0));
|
|
|
|
Tmp2 = PromoteOp(Node->getOperand(1));
|
|
|
|
assert(MVT::isInteger(NVT) && "Operators don't apply to FP!");
|
2005-04-13 10:38:47 +08:00
|
|
|
Tmp1 = DAG.getZeroExtendInReg(Tmp1, VT);
|
|
|
|
Tmp2 = DAG.getZeroExtendInReg(Tmp2, VT);
|
2005-01-16 06:16:26 +08:00
|
|
|
Result = DAG.getNode(Node->getOpcode(), NVT, Tmp1, Tmp2);
|
|
|
|
break;
|
|
|
|
|
|
|
|
case ISD::SHL:
|
|
|
|
Tmp1 = PromoteOp(Node->getOperand(0));
|
|
|
|
Tmp2 = LegalizeOp(Node->getOperand(1));
|
|
|
|
Result = DAG.getNode(ISD::SHL, NVT, Tmp1, Tmp2);
|
|
|
|
break;
|
|
|
|
case ISD::SRA:
|
|
|
|
// The input value must be properly sign extended.
|
|
|
|
Tmp1 = PromoteOp(Node->getOperand(0));
|
2005-07-10 08:07:11 +08:00
|
|
|
Tmp1 = DAG.getNode(ISD::SIGN_EXTEND_INREG, NVT, Tmp1,
|
|
|
|
DAG.getValueType(VT));
|
2005-01-16 06:16:26 +08:00
|
|
|
Tmp2 = LegalizeOp(Node->getOperand(1));
|
|
|
|
Result = DAG.getNode(ISD::SRA, NVT, Tmp1, Tmp2);
|
|
|
|
break;
|
|
|
|
case ISD::SRL:
|
|
|
|
// The input value must be properly zero extended.
|
|
|
|
Tmp1 = PromoteOp(Node->getOperand(0));
|
2005-04-13 10:38:47 +08:00
|
|
|
Tmp1 = DAG.getZeroExtendInReg(Tmp1, VT);
|
2005-01-16 06:16:26 +08:00
|
|
|
Tmp2 = LegalizeOp(Node->getOperand(1));
|
|
|
|
Result = DAG.getNode(ISD::SRL, NVT, Tmp1, Tmp2);
|
|
|
|
break;
|
2006-01-15 06:41:46 +08:00
|
|
|
|
2005-01-15 13:21:40 +08:00
|
|
|
case ISD::LOAD:
|
|
|
|
Tmp1 = LegalizeOp(Node->getOperand(0)); // Legalize the chain.
|
|
|
|
Tmp2 = LegalizeOp(Node->getOperand(1)); // Legalize the pointer.
|
2005-10-13 11:11:28 +08:00
|
|
|
Result = DAG.getExtLoad(ISD::EXTLOAD, NVT, Tmp1, Tmp2,
|
|
|
|
Node->getOperand(2), VT);
|
2005-01-15 13:21:40 +08:00
|
|
|
// Remember that we legalized the chain.
|
|
|
|
AddLegalizedOperand(Op.getValue(1), Result.getValue(1));
|
|
|
|
break;
|
2005-10-14 04:07:41 +08:00
|
|
|
case ISD::SEXTLOAD:
|
|
|
|
case ISD::ZEXTLOAD:
|
|
|
|
case ISD::EXTLOAD:
|
|
|
|
Tmp1 = LegalizeOp(Node->getOperand(0)); // Legalize the chain.
|
|
|
|
Tmp2 = LegalizeOp(Node->getOperand(1)); // Legalize the pointer.
|
2005-10-16 04:24:07 +08:00
|
|
|
Result = DAG.getExtLoad(Node->getOpcode(), NVT, Tmp1, Tmp2,
|
|
|
|
Node->getOperand(2),
|
|
|
|
cast<VTSDNode>(Node->getOperand(3))->getVT());
|
2005-10-14 04:07:41 +08:00
|
|
|
// Remember that we legalized the chain.
|
|
|
|
AddLegalizedOperand(Op.getValue(1), Result.getValue(1));
|
|
|
|
break;
|
2005-01-15 13:21:40 +08:00
|
|
|
case ISD::SELECT:
|
2005-01-19 03:27:06 +08:00
|
|
|
switch (getTypeAction(Node->getOperand(0).getValueType())) {
|
|
|
|
case Expand: assert(0 && "It's impossible to expand bools");
|
|
|
|
case Legal:
|
|
|
|
Tmp1 = LegalizeOp(Node->getOperand(0));// Legalize the condition.
|
|
|
|
break;
|
|
|
|
case Promote:
|
|
|
|
Tmp1 = PromoteOp(Node->getOperand(0)); // Promote the condition.
|
|
|
|
break;
|
|
|
|
}
|
2005-01-15 13:21:40 +08:00
|
|
|
Tmp2 = PromoteOp(Node->getOperand(1)); // Legalize the op0
|
|
|
|
Tmp3 = PromoteOp(Node->getOperand(2)); // Legalize the op1
|
|
|
|
Result = DAG.getNode(ISD::SELECT, NVT, Tmp1, Tmp2, Tmp3);
|
|
|
|
break;
|
2005-08-11 04:51:12 +08:00
|
|
|
case ISD::SELECT_CC:
|
|
|
|
Tmp2 = PromoteOp(Node->getOperand(2)); // True
|
|
|
|
Tmp3 = PromoteOp(Node->getOperand(3)); // False
|
|
|
|
Result = DAG.getNode(ISD::SELECT_CC, NVT, Node->getOperand(0),
|
|
|
|
Node->getOperand(1), Tmp2, Tmp3,
|
|
|
|
Node->getOperand(4));
|
|
|
|
break;
|
2005-05-14 02:43:43 +08:00
|
|
|
case ISD::TAILCALL:
|
2005-01-17 03:46:48 +08:00
|
|
|
case ISD::CALL: {
|
|
|
|
Tmp1 = LegalizeOp(Node->getOperand(0)); // Legalize the chain.
|
|
|
|
Tmp2 = LegalizeOp(Node->getOperand(1)); // Legalize the callee.
|
|
|
|
|
2005-01-20 04:24:35 +08:00
|
|
|
std::vector<SDOperand> Ops;
|
|
|
|
for (unsigned i = 2, e = Node->getNumOperands(); i != e; ++i)
|
|
|
|
Ops.push_back(LegalizeOp(Node->getOperand(i)));
|
|
|
|
|
2005-01-17 03:46:48 +08:00
|
|
|
assert(Node->getNumValues() == 2 && Op.ResNo == 0 &&
|
|
|
|
"Can only promote single result calls");
|
|
|
|
std::vector<MVT::ValueType> RetTyVTs;
|
|
|
|
RetTyVTs.reserve(2);
|
|
|
|
RetTyVTs.push_back(NVT);
|
|
|
|
RetTyVTs.push_back(MVT::Other);
|
2005-05-14 02:43:43 +08:00
|
|
|
SDNode *NC = DAG.getCall(RetTyVTs, Tmp1, Tmp2, Ops,
|
|
|
|
Node->getOpcode() == ISD::TAILCALL);
|
2005-01-17 03:46:48 +08:00
|
|
|
Result = SDOperand(NC, 0);
|
|
|
|
|
|
|
|
// Insert the new chain mapping.
|
|
|
|
AddLegalizedOperand(Op.getValue(1), Result.getValue(1));
|
|
|
|
break;
|
2005-04-22 06:36:52 +08:00
|
|
|
}
|
2006-01-14 11:14:10 +08:00
|
|
|
case ISD::BSWAP:
|
|
|
|
Tmp1 = Node->getOperand(0);
|
|
|
|
Tmp1 = DAG.getNode(ISD::ZERO_EXTEND, NVT, Tmp1);
|
|
|
|
Tmp1 = DAG.getNode(ISD::BSWAP, NVT, Tmp1);
|
|
|
|
Result = DAG.getNode(ISD::SRL, NVT, Tmp1,
|
|
|
|
DAG.getConstant(getSizeInBits(NVT) - getSizeInBits(VT),
|
|
|
|
TLI.getShiftAmountTy()));
|
|
|
|
break;
|
2005-05-05 03:11:05 +08:00
|
|
|
case ISD::CTPOP:
|
|
|
|
case ISD::CTTZ:
|
|
|
|
case ISD::CTLZ:
|
|
|
|
Tmp1 = Node->getOperand(0);
|
|
|
|
//Zero extend the argument
|
|
|
|
Tmp1 = DAG.getNode(ISD::ZERO_EXTEND, NVT, Tmp1);
|
|
|
|
// Perform the larger operation, then subtract if needed.
|
|
|
|
Tmp1 = DAG.getNode(Node->getOpcode(), NVT, Tmp1);
|
|
|
|
switch(Node->getOpcode())
|
|
|
|
{
|
|
|
|
case ISD::CTPOP:
|
|
|
|
Result = Tmp1;
|
|
|
|
break;
|
|
|
|
case ISD::CTTZ:
|
|
|
|
//if Tmp1 == sizeinbits(NVT) then Tmp1 = sizeinbits(Old VT)
|
Teach the legalizer how to legalize FP_TO_UINT.
Teach the legalizer to promote FP_TO_UINT to FP_TO_SINT if the wider
FP_TO_UINT is also illegal. This allows us on PPC to codegen
unsigned short foo(float a) { return a; }
as:
_foo:
.LBB_foo_0: ; entry
fctiwz f0, f1
stfd f0, -8(r1)
lwz r2, -4(r1)
rlwinm r3, r2, 0, 16, 31
blr
instead of:
_foo:
.LBB_foo_0: ; entry
fctiwz f0, f1
stfd f0, -8(r1)
lwz r2, -4(r1)
lis r3, ha16(.CPI_foo_0)
lfs f0, lo16(.CPI_foo_0)(r3)
fcmpu cr0, f1, f0
blt .LBB_foo_2 ; entry
.LBB_foo_1: ; entry
fsubs f0, f1, f0
fctiwz f0, f0
stfd f0, -16(r1)
lwz r2, -12(r1)
xoris r2, r2, 32768
.LBB_foo_2: ; entry
rlwinm r3, r2, 0, 16, 31
blr
llvm-svn: 22785
2005-08-14 09:20:53 +08:00
|
|
|
Tmp2 = DAG.getSetCC(TLI.getSetCCResultTy(), Tmp1,
|
2005-08-10 04:20:18 +08:00
|
|
|
DAG.getConstant(getSizeInBits(NVT), NVT), ISD::SETEQ);
|
2005-07-27 14:12:32 +08:00
|
|
|
Result = DAG.getNode(ISD::SELECT, NVT, Tmp2,
|
2005-05-05 03:11:05 +08:00
|
|
|
DAG.getConstant(getSizeInBits(VT),NVT), Tmp1);
|
|
|
|
break;
|
|
|
|
case ISD::CTLZ:
|
|
|
|
//Tmp1 = Tmp1 - (sizeinbits(NVT) - sizeinbits(Old VT))
|
2005-07-27 14:12:32 +08:00
|
|
|
Result = DAG.getNode(ISD::SUB, NVT, Tmp1,
|
|
|
|
DAG.getConstant(getSizeInBits(NVT) -
|
2005-05-05 03:11:05 +08:00
|
|
|
getSizeInBits(VT), NVT));
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
break;
|
2005-01-15 13:21:40 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
assert(Result.Val && "Didn't set a result!");
|
|
|
|
AddPromotedOperand(Op, Result);
|
|
|
|
return Result;
|
|
|
|
}
|
2005-01-07 15:47:09 +08:00
|
|
|
|
2005-12-23 08:16:34 +08:00
|
|
|
/// ExpandBIT_CONVERT - Expand a BIT_CONVERT node into a store/load combination.
|
2005-12-23 08:52:30 +08:00
|
|
|
/// The resultant code need not be legal. Note that SrcOp is the input operand
|
|
|
|
/// to the BIT_CONVERT, not the BIT_CONVERT node itself.
|
2005-12-23 08:16:34 +08:00
|
|
|
SDOperand SelectionDAGLegalize::ExpandBIT_CONVERT(MVT::ValueType DestVT,
|
|
|
|
SDOperand SrcOp) {
|
|
|
|
// Create the stack frame object.
|
|
|
|
MachineFrameInfo *FrameInfo = DAG.getMachineFunction().getFrameInfo();
|
|
|
|
unsigned ByteSize = MVT::getSizeInBits(DestVT)/8;
|
2005-12-23 08:52:30 +08:00
|
|
|
int FrameIdx = FrameInfo->CreateStackObject(ByteSize, ByteSize);
|
2005-12-23 08:16:34 +08:00
|
|
|
SDOperand FIPtr = DAG.getFrameIndex(FrameIdx, TLI.getPointerTy());
|
|
|
|
|
|
|
|
// Emit a store to the stack slot.
|
|
|
|
SDOperand Store = DAG.getNode(ISD::STORE, MVT::Other, DAG.getEntryNode(),
|
2005-12-23 08:50:25 +08:00
|
|
|
SrcOp, FIPtr, DAG.getSrcValue(NULL));
|
2005-12-23 08:16:34 +08:00
|
|
|
// Result is a load from the stack slot.
|
|
|
|
return DAG.getLoad(DestVT, Store, FIPtr, DAG.getSrcValue(0));
|
|
|
|
}
|
|
|
|
|
2005-01-21 02:52:28 +08:00
|
|
|
/// ExpandAddSub - Find a clever way to expand this add operation into
|
|
|
|
/// subcomponents.
|
2005-04-02 11:38:53 +08:00
|
|
|
void SelectionDAGLegalize::
|
|
|
|
ExpandByParts(unsigned NodeOp, SDOperand LHS, SDOperand RHS,
|
|
|
|
SDOperand &Lo, SDOperand &Hi) {
|
2005-01-21 02:52:28 +08:00
|
|
|
// Expand the subcomponents.
|
|
|
|
SDOperand LHSL, LHSH, RHSL, RHSH;
|
|
|
|
ExpandOp(LHS, LHSL, LHSH);
|
|
|
|
ExpandOp(RHS, RHSL, RHSH);
|
|
|
|
|
|
|
|
std::vector<SDOperand> Ops;
|
|
|
|
Ops.push_back(LHSL);
|
|
|
|
Ops.push_back(LHSH);
|
|
|
|
Ops.push_back(RHSL);
|
|
|
|
Ops.push_back(RHSH);
|
2005-05-14 15:25:05 +08:00
|
|
|
std::vector<MVT::ValueType> VTs(2, LHSL.getValueType());
|
|
|
|
Lo = DAG.getNode(NodeOp, VTs, Ops);
|
2005-01-21 02:52:28 +08:00
|
|
|
Hi = Lo.getValue(1);
|
|
|
|
}
|
|
|
|
|
2005-04-02 12:00:59 +08:00
|
|
|
void SelectionDAGLegalize::ExpandShiftParts(unsigned NodeOp,
|
|
|
|
SDOperand Op, SDOperand Amt,
|
|
|
|
SDOperand &Lo, SDOperand &Hi) {
|
|
|
|
// Expand the subcomponents.
|
|
|
|
SDOperand LHSL, LHSH;
|
|
|
|
ExpandOp(Op, LHSL, LHSH);
|
|
|
|
|
|
|
|
std::vector<SDOperand> Ops;
|
|
|
|
Ops.push_back(LHSL);
|
|
|
|
Ops.push_back(LHSH);
|
|
|
|
Ops.push_back(Amt);
|
2005-08-31 01:21:17 +08:00
|
|
|
std::vector<MVT::ValueType> VTs(2, LHSL.getValueType());
|
2005-05-14 15:25:05 +08:00
|
|
|
Lo = DAG.getNode(NodeOp, VTs, Ops);
|
2005-04-02 12:00:59 +08:00
|
|
|
Hi = Lo.getValue(1);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2005-01-19 12:19:40 +08:00
|
|
|
/// ExpandShift - Try to find a clever way to expand this shift operation out to
|
|
|
|
/// smaller elements. If we can't find a way that is more efficient than a
|
|
|
|
/// libcall on this target, return false. Otherwise, return true with the
|
|
|
|
/// low-parts expanded into Lo and Hi.
|
|
|
|
bool SelectionDAGLegalize::ExpandShift(unsigned Opc, SDOperand Op,SDOperand Amt,
|
|
|
|
SDOperand &Lo, SDOperand &Hi) {
|
|
|
|
assert((Opc == ISD::SHL || Opc == ISD::SRA || Opc == ISD::SRL) &&
|
|
|
|
"This is not a shift!");
|
2005-04-07 05:13:14 +08:00
|
|
|
|
2005-01-19 12:19:40 +08:00
|
|
|
MVT::ValueType NVT = TLI.getTypeToTransformTo(Op.getValueType());
|
2005-04-07 05:13:14 +08:00
|
|
|
SDOperand ShAmt = LegalizeOp(Amt);
|
|
|
|
MVT::ValueType ShTy = ShAmt.getValueType();
|
|
|
|
unsigned VTBits = MVT::getSizeInBits(Op.getValueType());
|
|
|
|
unsigned NVTBits = MVT::getSizeInBits(NVT);
|
|
|
|
|
|
|
|
// Handle the case when Amt is an immediate. Other cases are currently broken
|
|
|
|
// and are disabled.
|
|
|
|
if (ConstantSDNode *CN = dyn_cast<ConstantSDNode>(Amt.Val)) {
|
|
|
|
unsigned Cst = CN->getValue();
|
|
|
|
// Expand the incoming operand to be shifted, so that we have its parts
|
|
|
|
SDOperand InL, InH;
|
|
|
|
ExpandOp(Op, InL, InH);
|
|
|
|
switch(Opc) {
|
|
|
|
case ISD::SHL:
|
|
|
|
if (Cst > VTBits) {
|
|
|
|
Lo = DAG.getConstant(0, NVT);
|
|
|
|
Hi = DAG.getConstant(0, NVT);
|
|
|
|
} else if (Cst > NVTBits) {
|
|
|
|
Lo = DAG.getConstant(0, NVT);
|
|
|
|
Hi = DAG.getNode(ISD::SHL, NVT, InL, DAG.getConstant(Cst-NVTBits,ShTy));
|
2005-04-12 04:08:52 +08:00
|
|
|
} else if (Cst == NVTBits) {
|
|
|
|
Lo = DAG.getConstant(0, NVT);
|
|
|
|
Hi = InL;
|
2005-04-07 05:13:14 +08:00
|
|
|
} else {
|
|
|
|
Lo = DAG.getNode(ISD::SHL, NVT, InL, DAG.getConstant(Cst, ShTy));
|
|
|
|
Hi = DAG.getNode(ISD::OR, NVT,
|
|
|
|
DAG.getNode(ISD::SHL, NVT, InH, DAG.getConstant(Cst, ShTy)),
|
|
|
|
DAG.getNode(ISD::SRL, NVT, InL, DAG.getConstant(NVTBits-Cst, ShTy)));
|
|
|
|
}
|
|
|
|
return true;
|
|
|
|
case ISD::SRL:
|
|
|
|
if (Cst > VTBits) {
|
|
|
|
Lo = DAG.getConstant(0, NVT);
|
|
|
|
Hi = DAG.getConstant(0, NVT);
|
|
|
|
} else if (Cst > NVTBits) {
|
|
|
|
Lo = DAG.getNode(ISD::SRL, NVT, InH, DAG.getConstant(Cst-NVTBits,ShTy));
|
|
|
|
Hi = DAG.getConstant(0, NVT);
|
2005-04-12 04:08:52 +08:00
|
|
|
} else if (Cst == NVTBits) {
|
|
|
|
Lo = InH;
|
|
|
|
Hi = DAG.getConstant(0, NVT);
|
2005-04-07 05:13:14 +08:00
|
|
|
} else {
|
|
|
|
Lo = DAG.getNode(ISD::OR, NVT,
|
|
|
|
DAG.getNode(ISD::SRL, NVT, InL, DAG.getConstant(Cst, ShTy)),
|
|
|
|
DAG.getNode(ISD::SHL, NVT, InH, DAG.getConstant(NVTBits-Cst, ShTy)));
|
|
|
|
Hi = DAG.getNode(ISD::SRL, NVT, InH, DAG.getConstant(Cst, ShTy));
|
|
|
|
}
|
|
|
|
return true;
|
|
|
|
case ISD::SRA:
|
|
|
|
if (Cst > VTBits) {
|
2005-04-22 06:36:52 +08:00
|
|
|
Hi = Lo = DAG.getNode(ISD::SRA, NVT, InH,
|
2005-04-07 05:13:14 +08:00
|
|
|
DAG.getConstant(NVTBits-1, ShTy));
|
|
|
|
} else if (Cst > NVTBits) {
|
2005-04-22 06:36:52 +08:00
|
|
|
Lo = DAG.getNode(ISD::SRA, NVT, InH,
|
2005-04-07 05:13:14 +08:00
|
|
|
DAG.getConstant(Cst-NVTBits, ShTy));
|
2005-04-22 06:36:52 +08:00
|
|
|
Hi = DAG.getNode(ISD::SRA, NVT, InH,
|
2005-04-07 05:13:14 +08:00
|
|
|
DAG.getConstant(NVTBits-1, ShTy));
|
2005-04-12 04:08:52 +08:00
|
|
|
} else if (Cst == NVTBits) {
|
|
|
|
Lo = InH;
|
2005-04-22 06:36:52 +08:00
|
|
|
Hi = DAG.getNode(ISD::SRA, NVT, InH,
|
2005-04-12 04:08:52 +08:00
|
|
|
DAG.getConstant(NVTBits-1, ShTy));
|
2005-04-07 05:13:14 +08:00
|
|
|
} else {
|
|
|
|
Lo = DAG.getNode(ISD::OR, NVT,
|
|
|
|
DAG.getNode(ISD::SRL, NVT, InL, DAG.getConstant(Cst, ShTy)),
|
|
|
|
DAG.getNode(ISD::SHL, NVT, InH, DAG.getConstant(NVTBits-Cst, ShTy)));
|
|
|
|
Hi = DAG.getNode(ISD::SRA, NVT, InH, DAG.getConstant(Cst, ShTy));
|
|
|
|
}
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
// FIXME: The following code for expanding shifts using ISD::SELECT is buggy,
|
|
|
|
// so disable it for now. Currently targets are handling this via SHL_PARTS
|
|
|
|
// and friends.
|
|
|
|
return false;
|
2005-01-19 12:19:40 +08:00
|
|
|
|
|
|
|
// If we have an efficient select operation (or if the selects will all fold
|
|
|
|
// away), lower to some complex code, otherwise just emit the libcall.
|
2005-08-25 00:35:28 +08:00
|
|
|
if (!TLI.isOperationLegal(ISD::SELECT, NVT) && !isa<ConstantSDNode>(Amt))
|
2005-01-19 12:19:40 +08:00
|
|
|
return false;
|
|
|
|
|
|
|
|
SDOperand InL, InH;
|
|
|
|
ExpandOp(Op, InL, InH);
|
|
|
|
SDOperand NAmt = DAG.getNode(ISD::SUB, ShTy, // NAmt = 32-ShAmt
|
|
|
|
DAG.getConstant(NVTBits, ShTy), ShAmt);
|
|
|
|
|
2005-01-21 04:29:23 +08:00
|
|
|
// Compare the unmasked shift amount against 32.
|
2005-08-10 04:20:18 +08:00
|
|
|
SDOperand Cond = DAG.getSetCC(TLI.getSetCCResultTy(), ShAmt,
|
|
|
|
DAG.getConstant(NVTBits, ShTy), ISD::SETGE);
|
2005-01-21 04:29:23 +08:00
|
|
|
|
2005-01-19 12:19:40 +08:00
|
|
|
if (TLI.getShiftAmountFlavor() != TargetLowering::Mask) {
|
|
|
|
ShAmt = DAG.getNode(ISD::AND, ShTy, ShAmt, // ShAmt &= 31
|
|
|
|
DAG.getConstant(NVTBits-1, ShTy));
|
|
|
|
NAmt = DAG.getNode(ISD::AND, ShTy, NAmt, // NAmt &= 31
|
|
|
|
DAG.getConstant(NVTBits-1, ShTy));
|
|
|
|
}
|
|
|
|
|
|
|
|
if (Opc == ISD::SHL) {
|
|
|
|
SDOperand T1 = DAG.getNode(ISD::OR, NVT,// T1 = (Hi << Amt) | (Lo >> NAmt)
|
|
|
|
DAG.getNode(ISD::SHL, NVT, InH, ShAmt),
|
|
|
|
DAG.getNode(ISD::SRL, NVT, InL, NAmt));
|
2005-01-21 04:29:23 +08:00
|
|
|
SDOperand T2 = DAG.getNode(ISD::SHL, NVT, InL, ShAmt); // T2 = Lo << Amt&31
|
2005-04-22 06:36:52 +08:00
|
|
|
|
2005-01-19 12:19:40 +08:00
|
|
|
Hi = DAG.getNode(ISD::SELECT, NVT, Cond, T2, T1);
|
|
|
|
Lo = DAG.getNode(ISD::SELECT, NVT, Cond, DAG.getConstant(0, NVT), T2);
|
|
|
|
} else {
|
2005-01-21 14:05:23 +08:00
|
|
|
SDOperand HiLoPart = DAG.getNode(ISD::SELECT, NVT,
|
2005-08-10 04:20:18 +08:00
|
|
|
DAG.getSetCC(TLI.getSetCCResultTy(), NAmt,
|
|
|
|
DAG.getConstant(32, ShTy),
|
|
|
|
ISD::SETEQ),
|
2005-01-21 14:05:23 +08:00
|
|
|
DAG.getConstant(0, NVT),
|
|
|
|
DAG.getNode(ISD::SHL, NVT, InH, NAmt));
|
2005-01-19 12:19:40 +08:00
|
|
|
SDOperand T1 = DAG.getNode(ISD::OR, NVT,// T1 = (Hi << NAmt) | (Lo >> Amt)
|
2005-01-21 14:05:23 +08:00
|
|
|
HiLoPart,
|
2005-01-19 12:19:40 +08:00
|
|
|
DAG.getNode(ISD::SRL, NVT, InL, ShAmt));
|
2005-01-21 04:29:23 +08:00
|
|
|
SDOperand T2 = DAG.getNode(Opc, NVT, InH, ShAmt); // T2 = InH >> ShAmt&31
|
2005-01-19 12:19:40 +08:00
|
|
|
|
|
|
|
SDOperand HiPart;
|
2005-01-21 14:05:23 +08:00
|
|
|
if (Opc == ISD::SRA)
|
|
|
|
HiPart = DAG.getNode(ISD::SRA, NVT, InH,
|
|
|
|
DAG.getConstant(NVTBits-1, ShTy));
|
2005-01-19 12:19:40 +08:00
|
|
|
else
|
|
|
|
HiPart = DAG.getConstant(0, NVT);
|
|
|
|
Lo = DAG.getNode(ISD::SELECT, NVT, Cond, T2, T1);
|
2005-01-21 04:29:23 +08:00
|
|
|
Hi = DAG.getNode(ISD::SELECT, NVT, Cond, HiPart, T2);
|
2005-01-19 12:19:40 +08:00
|
|
|
}
|
|
|
|
return true;
|
|
|
|
}
|
2005-01-21 14:05:23 +08:00
|
|
|
|
2005-05-13 13:09:11 +08:00
|
|
|
/// FindLatestCallSeqStart - Scan up the dag to find the latest (highest
|
|
|
|
/// NodeDepth) node that is an CallSeqStart operation and occurs later than
|
2005-01-23 12:42:50 +08:00
|
|
|
/// Found.
|
2006-01-10 07:21:49 +08:00
|
|
|
static void FindLatestCallSeqStart(SDNode *Node, SDNode *&Found,
|
|
|
|
std::set<SDNode*> &Visited) {
|
2006-01-21 02:40:10 +08:00
|
|
|
if (Node->getNodeDepth() <= Found->getNodeDepth() ||
|
2006-01-10 07:21:49 +08:00
|
|
|
!Visited.insert(Node).second) return;
|
2005-08-06 00:23:57 +08:00
|
|
|
|
2005-05-13 07:24:06 +08:00
|
|
|
// If we found an CALLSEQ_START, we already know this node occurs later
|
2005-01-23 12:42:50 +08:00
|
|
|
// than the Found node. Just remember this node and return.
|
2005-05-13 07:24:06 +08:00
|
|
|
if (Node->getOpcode() == ISD::CALLSEQ_START) {
|
2005-01-23 12:42:50 +08:00
|
|
|
Found = Node;
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
// Otherwise, scan the operands of Node to see if any of them is a call.
|
|
|
|
assert(Node->getNumOperands() != 0 &&
|
|
|
|
"All leaves should have depth equal to the entry node!");
|
2005-10-06 05:44:10 +08:00
|
|
|
for (unsigned i = 0, e = Node->getNumOperands()-1; i != e; ++i)
|
2006-01-10 07:21:49 +08:00
|
|
|
FindLatestCallSeqStart(Node->getOperand(i).Val, Found, Visited);
|
2005-01-23 12:42:50 +08:00
|
|
|
|
|
|
|
// Tail recurse for the last iteration.
|
2005-05-13 13:09:11 +08:00
|
|
|
FindLatestCallSeqStart(Node->getOperand(Node->getNumOperands()-1).Val,
|
2006-01-10 07:21:49 +08:00
|
|
|
Found, Visited);
|
2005-01-23 12:42:50 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2005-05-13 13:09:11 +08:00
|
|
|
/// FindEarliestCallSeqEnd - Scan down the dag to find the earliest (lowest
|
|
|
|
/// NodeDepth) node that is an CallSeqEnd operation and occurs more recent
|
2005-01-23 12:42:50 +08:00
|
|
|
/// than Found.
|
2005-08-06 02:10:27 +08:00
|
|
|
static void FindEarliestCallSeqEnd(SDNode *Node, SDNode *&Found,
|
|
|
|
std::set<SDNode*> &Visited) {
|
2006-01-21 02:40:10 +08:00
|
|
|
if ((Found && Node->getNodeDepth() >= Found->getNodeDepth()) ||
|
2005-08-06 02:10:27 +08:00
|
|
|
!Visited.insert(Node).second) return;
|
2005-01-23 12:42:50 +08:00
|
|
|
|
2005-05-13 07:24:06 +08:00
|
|
|
// If we found an CALLSEQ_END, we already know this node occurs earlier
|
2005-01-23 12:42:50 +08:00
|
|
|
// than the Found node. Just remember this node and return.
|
2005-05-13 07:24:06 +08:00
|
|
|
if (Node->getOpcode() == ISD::CALLSEQ_END) {
|
2005-01-23 12:42:50 +08:00
|
|
|
Found = Node;
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
// Otherwise, scan the operands of Node to see if any of them is a call.
|
|
|
|
SDNode::use_iterator UI = Node->use_begin(), E = Node->use_end();
|
|
|
|
if (UI == E) return;
|
|
|
|
for (--E; UI != E; ++UI)
|
2005-08-06 02:10:27 +08:00
|
|
|
FindEarliestCallSeqEnd(*UI, Found, Visited);
|
2005-01-23 12:42:50 +08:00
|
|
|
|
|
|
|
// Tail recurse for the last iteration.
|
2005-08-06 02:10:27 +08:00
|
|
|
FindEarliestCallSeqEnd(*UI, Found, Visited);
|
2005-01-23 12:42:50 +08:00
|
|
|
}
|
|
|
|
|
2005-05-13 13:09:11 +08:00
|
|
|
/// FindCallSeqEnd - Given a chained node that is part of a call sequence,
|
2005-05-13 07:24:06 +08:00
|
|
|
/// find the CALLSEQ_END node that terminates the call sequence.
|
2005-05-13 13:09:11 +08:00
|
|
|
static SDNode *FindCallSeqEnd(SDNode *Node) {
|
2005-05-13 07:24:06 +08:00
|
|
|
if (Node->getOpcode() == ISD::CALLSEQ_END)
|
2005-01-23 12:42:50 +08:00
|
|
|
return Node;
|
2005-04-02 11:22:40 +08:00
|
|
|
if (Node->use_empty())
|
2005-05-13 13:09:11 +08:00
|
|
|
return 0; // No CallSeqEnd
|
2005-01-23 12:42:50 +08:00
|
|
|
|
2006-01-15 06:41:46 +08:00
|
|
|
// The chain is usually at the end.
|
2005-01-23 12:42:50 +08:00
|
|
|
SDOperand TheChain(Node, Node->getNumValues()-1);
|
2006-01-15 06:41:46 +08:00
|
|
|
if (TheChain.getValueType() != MVT::Other) {
|
|
|
|
// Sometimes it's at the beginning.
|
2005-05-14 16:34:53 +08:00
|
|
|
TheChain = SDOperand(Node, 0);
|
2006-01-15 06:41:46 +08:00
|
|
|
if (TheChain.getValueType() != MVT::Other) {
|
|
|
|
// Otherwise, hunt for it.
|
|
|
|
for (unsigned i = 1, e = Node->getNumValues(); i != e; ++i)
|
|
|
|
if (Node->getValueType(i) == MVT::Other) {
|
|
|
|
TheChain = SDOperand(Node, i);
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
|
|
|
// Otherwise, we walked into a node without a chain.
|
|
|
|
if (TheChain.getValueType() != MVT::Other)
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
}
|
2005-04-22 06:36:52 +08:00
|
|
|
|
|
|
|
for (SDNode::use_iterator UI = Node->use_begin(),
|
2005-08-06 00:23:57 +08:00
|
|
|
E = Node->use_end(); UI != E; ++UI) {
|
2005-04-22 06:36:52 +08:00
|
|
|
|
2005-01-23 12:42:50 +08:00
|
|
|
// Make sure to only follow users of our token chain.
|
|
|
|
SDNode *User = *UI;
|
|
|
|
for (unsigned i = 0, e = User->getNumOperands(); i != e; ++i)
|
|
|
|
if (User->getOperand(i) == TheChain)
|
2005-05-13 13:17:00 +08:00
|
|
|
if (SDNode *Result = FindCallSeqEnd(User))
|
|
|
|
return Result;
|
2005-01-23 12:42:50 +08:00
|
|
|
}
|
2005-08-06 00:23:57 +08:00
|
|
|
return 0;
|
2005-01-23 12:42:50 +08:00
|
|
|
}
|
|
|
|
|
2005-05-13 13:09:11 +08:00
|
|
|
/// FindCallSeqStart - Given a chained node that is part of a call sequence,
|
2005-05-13 07:24:06 +08:00
|
|
|
/// find the CALLSEQ_START node that initiates the call sequence.
|
2005-05-13 13:09:11 +08:00
|
|
|
static SDNode *FindCallSeqStart(SDNode *Node) {
|
|
|
|
assert(Node && "Didn't find callseq_start for a call??");
|
2005-05-13 07:24:06 +08:00
|
|
|
if (Node->getOpcode() == ISD::CALLSEQ_START) return Node;
|
2005-05-12 03:02:11 +08:00
|
|
|
|
|
|
|
assert(Node->getOperand(0).getValueType() == MVT::Other &&
|
|
|
|
"Node doesn't have a token chain argument!");
|
2005-05-13 13:09:11 +08:00
|
|
|
return FindCallSeqStart(Node->getOperand(0).Val);
|
2005-05-12 03:02:11 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2005-01-23 12:42:50 +08:00
|
|
|
/// FindInputOutputChains - If we are replacing an operation with a call we need
|
|
|
|
/// to find the call that occurs before and the call that occurs after it to
|
2005-05-12 03:02:11 +08:00
|
|
|
/// properly serialize the calls in the block. The returned operand is the
|
|
|
|
/// input chain value for the new call (e.g. the entry node or the previous
|
|
|
|
/// call), and OutChain is set to be the chain node to update to point to the
|
|
|
|
/// end of the call chain.
|
2005-01-23 12:42:50 +08:00
|
|
|
static SDOperand FindInputOutputChains(SDNode *OpNode, SDNode *&OutChain,
|
|
|
|
SDOperand Entry) {
|
2005-05-13 13:09:11 +08:00
|
|
|
SDNode *LatestCallSeqStart = Entry.Val;
|
|
|
|
SDNode *LatestCallSeqEnd = 0;
|
2006-01-10 07:21:49 +08:00
|
|
|
std::set<SDNode*> Visited;
|
|
|
|
FindLatestCallSeqStart(OpNode, LatestCallSeqStart, Visited);
|
|
|
|
Visited.clear();
|
2005-05-13 13:09:11 +08:00
|
|
|
//std::cerr<<"Found node: "; LatestCallSeqStart->dump(); std::cerr <<"\n";
|
2005-04-22 06:36:52 +08:00
|
|
|
|
2005-05-13 07:24:06 +08:00
|
|
|
// It is possible that no ISD::CALLSEQ_START was found because there is no
|
2005-04-11 11:01:51 +08:00
|
|
|
// previous call in the function. LatestCallStackDown may in that case be
|
2005-05-13 07:24:06 +08:00
|
|
|
// the entry node itself. Do not attempt to find a matching CALLSEQ_END
|
|
|
|
// unless LatestCallStackDown is an CALLSEQ_START.
|
2005-10-04 10:10:55 +08:00
|
|
|
if (LatestCallSeqStart->getOpcode() == ISD::CALLSEQ_START) {
|
2005-05-13 13:09:11 +08:00
|
|
|
LatestCallSeqEnd = FindCallSeqEnd(LatestCallSeqStart);
|
2005-10-04 10:10:55 +08:00
|
|
|
//std::cerr<<"Found end node: "; LatestCallSeqEnd->dump(); std::cerr <<"\n";
|
|
|
|
} else {
|
2005-05-13 13:09:11 +08:00
|
|
|
LatestCallSeqEnd = Entry.Val;
|
2005-10-04 10:10:55 +08:00
|
|
|
}
|
2005-05-13 13:09:11 +08:00
|
|
|
assert(LatestCallSeqEnd && "NULL return from FindCallSeqEnd");
|
2005-04-22 06:36:52 +08:00
|
|
|
|
2005-05-12 03:02:11 +08:00
|
|
|
// Finally, find the first call that this must come before, first we find the
|
2005-05-13 13:09:11 +08:00
|
|
|
// CallSeqEnd that ends the call.
|
2005-05-12 03:02:11 +08:00
|
|
|
OutChain = 0;
|
2005-08-06 02:10:27 +08:00
|
|
|
FindEarliestCallSeqEnd(OpNode, OutChain, Visited);
|
2006-01-10 07:21:49 +08:00
|
|
|
Visited.clear();
|
2005-01-23 12:42:50 +08:00
|
|
|
|
2005-05-13 13:09:11 +08:00
|
|
|
// If we found one, translate from the adj up to the callseq_start.
|
2005-05-12 03:02:11 +08:00
|
|
|
if (OutChain)
|
2005-05-13 13:09:11 +08:00
|
|
|
OutChain = FindCallSeqStart(OutChain);
|
2005-01-23 12:42:50 +08:00
|
|
|
|
2005-05-13 13:09:11 +08:00
|
|
|
return SDOperand(LatestCallSeqEnd, 0);
|
2005-01-23 12:42:50 +08:00
|
|
|
}
|
|
|
|
|
2005-07-27 14:12:32 +08:00
|
|
|
/// SpliceCallInto - Given the result chain of a libcall (CallResult), and a
|
2005-05-12 12:49:08 +08:00
|
|
|
void SelectionDAGLegalize::SpliceCallInto(const SDOperand &CallResult,
|
|
|
|
SDNode *OutChain) {
|
2005-05-12 03:02:11 +08:00
|
|
|
// Nothing to splice it into?
|
|
|
|
if (OutChain == 0) return;
|
|
|
|
|
|
|
|
assert(OutChain->getOperand(0).getValueType() == MVT::Other);
|
|
|
|
//OutChain->dump();
|
|
|
|
|
|
|
|
// Form a token factor node merging the old inval and the new inval.
|
|
|
|
SDOperand InToken = DAG.getNode(ISD::TokenFactor, MVT::Other, CallResult,
|
|
|
|
OutChain->getOperand(0));
|
|
|
|
// Change the node to refer to the new token.
|
|
|
|
OutChain->setAdjCallChain(InToken);
|
|
|
|
}
|
2005-01-23 12:42:50 +08:00
|
|
|
|
|
|
|
|
2005-01-21 14:05:23 +08:00
|
|
|
// ExpandLibCall - Expand a node into a call to a libcall. If the result value
|
|
|
|
// does not fit into a register, return the lo part and set the hi part to the
|
|
|
|
// by-reg argument. If it does fit into a single register, return the result
|
|
|
|
// and leave the Hi part unset.
|
|
|
|
SDOperand SelectionDAGLegalize::ExpandLibCall(const char *Name, SDNode *Node,
|
|
|
|
SDOperand &Hi) {
|
2005-01-23 12:42:50 +08:00
|
|
|
SDNode *OutChain;
|
|
|
|
SDOperand InChain = FindInputOutputChains(Node, OutChain,
|
|
|
|
DAG.getEntryNode());
|
2005-04-02 11:22:40 +08:00
|
|
|
if (InChain.Val == 0)
|
|
|
|
InChain = DAG.getEntryNode();
|
2005-01-23 12:42:50 +08:00
|
|
|
|
2005-01-21 14:05:23 +08:00
|
|
|
TargetLowering::ArgListTy Args;
|
|
|
|
for (unsigned i = 0, e = Node->getNumOperands(); i != e; ++i) {
|
|
|
|
MVT::ValueType ArgVT = Node->getOperand(i).getValueType();
|
|
|
|
const Type *ArgTy = MVT::getTypeForValueType(ArgVT);
|
|
|
|
Args.push_back(std::make_pair(Node->getOperand(i), ArgTy));
|
|
|
|
}
|
|
|
|
SDOperand Callee = DAG.getExternalSymbol(Name, TLI.getPointerTy());
|
2005-04-22 06:36:52 +08:00
|
|
|
|
2005-05-12 03:02:11 +08:00
|
|
|
// Splice the libcall in wherever FindInputOutputChains tells us to.
|
2005-01-21 14:05:23 +08:00
|
|
|
const Type *RetTy = MVT::getTypeForValueType(Node->getValueType(0));
|
2005-05-12 03:02:11 +08:00
|
|
|
std::pair<SDOperand,SDOperand> CallInfo =
|
2005-05-14 02:50:42 +08:00
|
|
|
TLI.LowerCallTo(InChain, RetTy, false, CallingConv::C, false,
|
|
|
|
Callee, Args, DAG);
|
2005-05-12 03:02:11 +08:00
|
|
|
|
2005-09-03 04:26:58 +08:00
|
|
|
SDOperand Result;
|
2005-05-12 03:02:11 +08:00
|
|
|
switch (getTypeAction(CallInfo.first.getValueType())) {
|
2005-01-21 14:05:23 +08:00
|
|
|
default: assert(0 && "Unknown thing");
|
|
|
|
case Legal:
|
2005-09-03 04:26:58 +08:00
|
|
|
Result = CallInfo.first;
|
|
|
|
break;
|
2005-01-21 14:05:23 +08:00
|
|
|
case Promote:
|
|
|
|
assert(0 && "Cannot promote this yet!");
|
|
|
|
case Expand:
|
2005-09-03 04:26:58 +08:00
|
|
|
ExpandOp(CallInfo.first, Result, Hi);
|
|
|
|
CallInfo.second = LegalizeOp(CallInfo.second);
|
|
|
|
break;
|
2005-01-21 14:05:23 +08:00
|
|
|
}
|
2005-09-03 04:26:58 +08:00
|
|
|
|
|
|
|
SpliceCallInto(CallInfo.second, OutChain);
|
|
|
|
NeedsAnotherIteration = true;
|
|
|
|
return Result;
|
2005-01-21 14:05:23 +08:00
|
|
|
}
|
|
|
|
|
2005-01-23 12:42:50 +08:00
|
|
|
|
2005-01-21 14:05:23 +08:00
|
|
|
/// ExpandIntToFP - Expand a [US]INT_TO_FP operation, assuming that the
|
|
|
|
/// destination type is legal.
|
|
|
|
SDOperand SelectionDAGLegalize::
|
|
|
|
ExpandIntToFP(bool isSigned, MVT::ValueType DestTy, SDOperand Source) {
|
2005-08-25 00:35:28 +08:00
|
|
|
assert(isTypeLegal(DestTy) && "Destination type is not legal!");
|
2005-01-21 14:05:23 +08:00
|
|
|
assert(getTypeAction(Source.getValueType()) == Expand &&
|
|
|
|
"This is not an expansion!");
|
|
|
|
assert(Source.getValueType() == MVT::i64 && "Only handle expand from i64!");
|
|
|
|
|
2005-05-12 03:02:11 +08:00
|
|
|
if (!isSigned) {
|
2005-04-13 13:09:42 +08:00
|
|
|
assert(Source.getValueType() == MVT::i64 &&
|
|
|
|
"This only works for 64-bit -> FP");
|
|
|
|
// The 64-bit value loaded will be incorrectly if the 'sign bit' of the
|
|
|
|
// incoming integer is set. To handle this, we dynamically test to see if
|
|
|
|
// it is set, and, if so, add a fudge factor.
|
|
|
|
SDOperand Lo, Hi;
|
|
|
|
ExpandOp(Source, Lo, Hi);
|
|
|
|
|
2005-05-13 12:45:13 +08:00
|
|
|
// If this is unsigned, and not supported, first perform the conversion to
|
|
|
|
// signed, then adjust the result if the sign bit is set.
|
|
|
|
SDOperand SignedConv = ExpandIntToFP(true, DestTy,
|
|
|
|
DAG.getNode(ISD::BUILD_PAIR, Source.getValueType(), Lo, Hi));
|
|
|
|
|
2005-08-10 04:20:18 +08:00
|
|
|
SDOperand SignSet = DAG.getSetCC(TLI.getSetCCResultTy(), Hi,
|
|
|
|
DAG.getConstant(0, Hi.getValueType()),
|
|
|
|
ISD::SETLT);
|
2005-04-13 13:09:42 +08:00
|
|
|
SDOperand Zero = getIntPtrConstant(0), Four = getIntPtrConstant(4);
|
|
|
|
SDOperand CstOffset = DAG.getNode(ISD::SELECT, Zero.getValueType(),
|
|
|
|
SignSet, Four, Zero);
|
2005-05-13 02:52:34 +08:00
|
|
|
uint64_t FF = 0x5f800000ULL;
|
|
|
|
if (TLI.isLittleEndian()) FF <<= 32;
|
|
|
|
static Constant *FudgeFactor = ConstantUInt::get(Type::ULongTy, FF);
|
2005-04-13 13:09:42 +08:00
|
|
|
|
2005-08-27 01:15:30 +08:00
|
|
|
SDOperand CPIdx = DAG.getConstantPool(FudgeFactor, TLI.getPointerTy());
|
2005-04-13 13:09:42 +08:00
|
|
|
CPIdx = DAG.getNode(ISD::ADD, TLI.getPointerTy(), CPIdx, CstOffset);
|
|
|
|
SDOperand FudgeInReg;
|
|
|
|
if (DestTy == MVT::f32)
|
2005-05-10 04:23:03 +08:00
|
|
|
FudgeInReg = DAG.getLoad(MVT::f32, DAG.getEntryNode(), CPIdx,
|
|
|
|
DAG.getSrcValue(NULL));
|
2005-04-13 13:09:42 +08:00
|
|
|
else {
|
|
|
|
assert(DestTy == MVT::f64 && "Unexpected conversion");
|
2005-07-10 09:55:33 +08:00
|
|
|
FudgeInReg = DAG.getExtLoad(ISD::EXTLOAD, MVT::f64, DAG.getEntryNode(),
|
|
|
|
CPIdx, DAG.getSrcValue(NULL), MVT::f32);
|
2005-04-13 13:09:42 +08:00
|
|
|
}
|
2005-09-29 14:44:39 +08:00
|
|
|
return DAG.getNode(ISD::FADD, DestTy, SignedConv, FudgeInReg);
|
2005-01-21 14:05:23 +08:00
|
|
|
}
|
2005-05-12 03:02:11 +08:00
|
|
|
|
2005-05-14 13:33:54 +08:00
|
|
|
// Check to see if the target has a custom way to lower this. If so, use it.
|
|
|
|
switch (TLI.getOperationAction(ISD::SINT_TO_FP, Source.getValueType())) {
|
|
|
|
default: assert(0 && "This action not implemented for this operation!");
|
|
|
|
case TargetLowering::Legal:
|
|
|
|
case TargetLowering::Expand:
|
|
|
|
break; // This case is handled below.
|
2005-08-26 08:14:16 +08:00
|
|
|
case TargetLowering::Custom: {
|
|
|
|
SDOperand NV = TLI.LowerOperation(DAG.getNode(ISD::SINT_TO_FP, DestTy,
|
|
|
|
Source), DAG);
|
|
|
|
if (NV.Val)
|
|
|
|
return LegalizeOp(NV);
|
|
|
|
break; // The target decided this was legal after all
|
|
|
|
}
|
2005-05-14 13:33:54 +08:00
|
|
|
}
|
|
|
|
|
2005-05-12 15:00:44 +08:00
|
|
|
// Expand the source, then glue it back together for the call. We must expand
|
|
|
|
// the source in case it is shared (this pass of legalize must traverse it).
|
|
|
|
SDOperand SrcLo, SrcHi;
|
|
|
|
ExpandOp(Source, SrcLo, SrcHi);
|
|
|
|
Source = DAG.getNode(ISD::BUILD_PAIR, Source.getValueType(), SrcLo, SrcHi);
|
|
|
|
|
2005-05-12 03:02:11 +08:00
|
|
|
SDNode *OutChain = 0;
|
|
|
|
SDOperand InChain = FindInputOutputChains(Source.Val, OutChain,
|
|
|
|
DAG.getEntryNode());
|
|
|
|
const char *FnName = 0;
|
|
|
|
if (DestTy == MVT::f32)
|
|
|
|
FnName = "__floatdisf";
|
|
|
|
else {
|
|
|
|
assert(DestTy == MVT::f64 && "Unknown fp value type!");
|
|
|
|
FnName = "__floatdidf";
|
|
|
|
}
|
|
|
|
|
2005-01-21 14:05:23 +08:00
|
|
|
SDOperand Callee = DAG.getExternalSymbol(FnName, TLI.getPointerTy());
|
|
|
|
|
|
|
|
TargetLowering::ArgListTy Args;
|
|
|
|
const Type *ArgTy = MVT::getTypeForValueType(Source.getValueType());
|
2005-05-12 14:54:21 +08:00
|
|
|
|
2005-01-21 14:05:23 +08:00
|
|
|
Args.push_back(std::make_pair(Source, ArgTy));
|
|
|
|
|
|
|
|
// We don't care about token chains for libcalls. We just use the entry
|
|
|
|
// node as our input and ignore the output chain. This allows us to place
|
|
|
|
// calls wherever we need them to satisfy data dependences.
|
|
|
|
const Type *RetTy = MVT::getTypeForValueType(DestTy);
|
2005-05-12 03:02:11 +08:00
|
|
|
|
|
|
|
std::pair<SDOperand,SDOperand> CallResult =
|
2005-05-14 02:50:42 +08:00
|
|
|
TLI.LowerCallTo(InChain, RetTy, false, CallingConv::C, true,
|
|
|
|
Callee, Args, DAG);
|
2005-05-12 03:02:11 +08:00
|
|
|
|
2005-05-12 12:49:08 +08:00
|
|
|
SpliceCallInto(CallResult.second, OutChain);
|
2005-05-12 03:02:11 +08:00
|
|
|
return CallResult.first;
|
2005-01-21 14:05:23 +08:00
|
|
|
}
|
2005-04-22 06:36:52 +08:00
|
|
|
|
2005-01-19 12:19:40 +08:00
|
|
|
|
|
|
|
|
2005-01-07 15:47:09 +08:00
|
|
|
/// ExpandOp - Expand the specified SDOperand into its two component pieces
|
|
|
|
/// Lo&Hi. Note that the Op MUST be an expanded type. As a result of this, the
|
|
|
|
/// LegalizeNodes map is filled in for any results that are not expanded, the
|
|
|
|
/// ExpandedNodes map is filled in for any results that are expanded, and the
|
|
|
|
/// Lo/Hi values are returned.
|
|
|
|
void SelectionDAGLegalize::ExpandOp(SDOperand Op, SDOperand &Lo, SDOperand &Hi){
|
|
|
|
MVT::ValueType VT = Op.getValueType();
|
2005-01-16 09:11:45 +08:00
|
|
|
MVT::ValueType NVT = TLI.getTypeToTransformTo(VT);
|
2005-01-07 15:47:09 +08:00
|
|
|
SDNode *Node = Op.Val;
|
|
|
|
assert(getTypeAction(VT) == Expand && "Not an expanded type!");
|
Check in code to scalarize arbitrarily wide packed types for some simple
vector operations (load, add, sub, mul).
This allows us to codegen:
void %foo(<4 x float> * %a) {
entry:
%tmp1 = load <4 x float> * %a;
%tmp2 = add <4 x float> %tmp1, %tmp1
store <4 x float> %tmp2, <4 x float> *%a
ret void
}
on ppc as:
_foo:
lfs f0, 12(r3)
lfs f1, 8(r3)
lfs f2, 4(r3)
lfs f3, 0(r3)
fadds f0, f0, f0
fadds f1, f1, f1
fadds f2, f2, f2
fadds f3, f3, f3
stfs f0, 12(r3)
stfs f1, 8(r3)
stfs f2, 4(r3)
stfs f3, 0(r3)
blr
llvm-svn: 24484
2005-11-23 02:16:00 +08:00
|
|
|
assert((MVT::isInteger(VT) || VT == MVT::Vector) &&
|
|
|
|
"Cannot expand FP values!");
|
|
|
|
assert(((MVT::isInteger(NVT) && NVT < VT) || VT == MVT::Vector) &&
|
2005-01-07 15:47:09 +08:00
|
|
|
"Cannot expand to FP value or to larger int value!");
|
|
|
|
|
2005-09-03 04:32:45 +08:00
|
|
|
// See if we already expanded it.
|
|
|
|
std::map<SDOperand, std::pair<SDOperand, SDOperand> >::iterator I
|
|
|
|
= ExpandedNodes.find(Op);
|
|
|
|
if (I != ExpandedNodes.end()) {
|
|
|
|
Lo = I->second.first;
|
|
|
|
Hi = I->second.second;
|
|
|
|
return;
|
2005-01-07 15:47:09 +08:00
|
|
|
}
|
|
|
|
|
2005-01-09 03:27:05 +08:00
|
|
|
// Expanding to multiple registers needs to perform an optimization step, and
|
|
|
|
// is not careful to avoid operations the target does not support. Make sure
|
|
|
|
// that all generated operations are legalized in the next iteration.
|
|
|
|
NeedsAnotherIteration = true;
|
2005-01-07 15:47:09 +08:00
|
|
|
|
|
|
|
switch (Node->getOpcode()) {
|
2006-01-21 12:27:00 +08:00
|
|
|
case ISD::CopyFromReg:
|
|
|
|
assert(0 && "CopyFromReg must be legal!");
|
|
|
|
default:
|
2005-01-07 15:47:09 +08:00
|
|
|
std::cerr << "NODE: "; Node->dump(); std::cerr << "\n";
|
|
|
|
assert(0 && "Do not know how to expand this operator!");
|
|
|
|
abort();
|
2005-04-02 06:34:39 +08:00
|
|
|
case ISD::UNDEF:
|
|
|
|
Lo = DAG.getNode(ISD::UNDEF, NVT);
|
|
|
|
Hi = DAG.getNode(ISD::UNDEF, NVT);
|
|
|
|
break;
|
2005-01-07 15:47:09 +08:00
|
|
|
case ISD::Constant: {
|
|
|
|
uint64_t Cst = cast<ConstantSDNode>(Node)->getValue();
|
|
|
|
Lo = DAG.getConstant(Cst, NVT);
|
|
|
|
Hi = DAG.getConstant(Cst >> MVT::getSizeInBits(NVT), NVT);
|
|
|
|
break;
|
|
|
|
}
|
Fix a crash where ConstantVec nodes were being generated with the wrong
type when the target did not support them. Also teach Legalize how to
expand ConstantVecs.
This allows us to generate
_test:
lwz r2, 12(r3)
lwz r4, 8(r3)
lwz r5, 4(r3)
lwz r6, 0(r3)
addi r2, r2, 4
addi r4, r4, 3
addi r5, r5, 2
addi r6, r6, 1
stw r2, 12(r3)
stw r4, 8(r3)
stw r5, 4(r3)
stw r6, 0(r3)
blr
For:
void %test(%v4i *%P) {
%T = load %v4i* %P
%S = add %v4i %T, <int 1, int 2, int 3, int 4>
store %v4i %S, %v4i * %P
ret void
}
On PowerPC.
llvm-svn: 24633
2005-12-08 03:48:11 +08:00
|
|
|
case ISD::ConstantVec: {
|
|
|
|
unsigned NumElements = Node->getNumOperands();
|
|
|
|
// If we only have two elements left in the constant vector, just break it
|
|
|
|
// apart into the two scalar constants it contains. Otherwise, bisect the
|
|
|
|
// ConstantVec, and return each half as a new ConstantVec.
|
|
|
|
// FIXME: this is hard coded as big endian, it may have to change to support
|
|
|
|
// SSE and Alpha MVI
|
|
|
|
if (NumElements == 2) {
|
|
|
|
Hi = Node->getOperand(0);
|
|
|
|
Lo = Node->getOperand(1);
|
|
|
|
} else {
|
|
|
|
NumElements /= 2;
|
|
|
|
std::vector<SDOperand> LoOps, HiOps;
|
|
|
|
for (unsigned I = 0, E = NumElements; I < E; ++I) {
|
|
|
|
HiOps.push_back(Node->getOperand(I));
|
|
|
|
LoOps.push_back(Node->getOperand(I+NumElements));
|
|
|
|
}
|
|
|
|
Lo = DAG.getNode(ISD::ConstantVec, MVT::Vector, LoOps);
|
|
|
|
Hi = DAG.getNode(ISD::ConstantVec, MVT::Vector, HiOps);
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
}
|
2005-01-07 15:47:09 +08:00
|
|
|
|
2005-03-29 06:03:13 +08:00
|
|
|
case ISD::BUILD_PAIR:
|
|
|
|
// Legalize both operands. FIXME: in the future we should handle the case
|
|
|
|
// where the two elements are not legal.
|
|
|
|
assert(isTypeLegal(NVT) && "Cannot expand this multiple times yet!");
|
|
|
|
Lo = LegalizeOp(Node->getOperand(0));
|
|
|
|
Hi = LegalizeOp(Node->getOperand(1));
|
|
|
|
break;
|
2005-12-13 06:27:43 +08:00
|
|
|
|
|
|
|
case ISD::SIGN_EXTEND_INREG:
|
|
|
|
ExpandOp(Node->getOperand(0), Lo, Hi);
|
|
|
|
// Sign extend the lo-part.
|
|
|
|
Hi = DAG.getNode(ISD::SRA, NVT, Lo,
|
|
|
|
DAG.getConstant(MVT::getSizeInBits(NVT)-1,
|
|
|
|
TLI.getShiftAmountTy()));
|
|
|
|
// sext_inreg the low part if needed.
|
|
|
|
Lo = DAG.getNode(ISD::SIGN_EXTEND_INREG, NVT, Lo, Node->getOperand(1));
|
|
|
|
break;
|
2005-03-29 06:03:13 +08:00
|
|
|
|
2006-01-14 11:14:10 +08:00
|
|
|
case ISD::BSWAP: {
|
|
|
|
ExpandOp(Node->getOperand(0), Lo, Hi);
|
|
|
|
SDOperand TempLo = DAG.getNode(ISD::BSWAP, NVT, Hi);
|
|
|
|
Hi = DAG.getNode(ISD::BSWAP, NVT, Lo);
|
|
|
|
Lo = TempLo;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
2005-05-11 12:51:16 +08:00
|
|
|
case ISD::CTPOP:
|
|
|
|
ExpandOp(Node->getOperand(0), Lo, Hi);
|
2005-05-11 13:09:47 +08:00
|
|
|
Lo = DAG.getNode(ISD::ADD, NVT, // ctpop(HL) -> ctpop(H)+ctpop(L)
|
|
|
|
DAG.getNode(ISD::CTPOP, NVT, Lo),
|
|
|
|
DAG.getNode(ISD::CTPOP, NVT, Hi));
|
2005-05-11 12:51:16 +08:00
|
|
|
Hi = DAG.getConstant(0, NVT);
|
|
|
|
break;
|
|
|
|
|
2005-05-13 03:05:01 +08:00
|
|
|
case ISD::CTLZ: {
|
|
|
|
// ctlz (HL) -> ctlz(H) != 32 ? ctlz(H) : (ctlz(L)+32)
|
2005-05-13 03:27:51 +08:00
|
|
|
ExpandOp(Node->getOperand(0), Lo, Hi);
|
2005-05-13 03:05:01 +08:00
|
|
|
SDOperand BitsC = DAG.getConstant(MVT::getSizeInBits(NVT), NVT);
|
|
|
|
SDOperand HLZ = DAG.getNode(ISD::CTLZ, NVT, Hi);
|
2005-08-10 04:20:18 +08:00
|
|
|
SDOperand TopNotZero = DAG.getSetCC(TLI.getSetCCResultTy(), HLZ, BitsC,
|
|
|
|
ISD::SETNE);
|
2005-05-13 03:05:01 +08:00
|
|
|
SDOperand LowPart = DAG.getNode(ISD::CTLZ, NVT, Lo);
|
|
|
|
LowPart = DAG.getNode(ISD::ADD, NVT, LowPart, BitsC);
|
|
|
|
|
|
|
|
Lo = DAG.getNode(ISD::SELECT, NVT, TopNotZero, HLZ, LowPart);
|
|
|
|
Hi = DAG.getConstant(0, NVT);
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
|
|
|
case ISD::CTTZ: {
|
|
|
|
// cttz (HL) -> cttz(L) != 32 ? cttz(L) : (cttz(H)+32)
|
2005-05-13 03:27:51 +08:00
|
|
|
ExpandOp(Node->getOperand(0), Lo, Hi);
|
2005-05-13 03:05:01 +08:00
|
|
|
SDOperand BitsC = DAG.getConstant(MVT::getSizeInBits(NVT), NVT);
|
|
|
|
SDOperand LTZ = DAG.getNode(ISD::CTTZ, NVT, Lo);
|
2005-08-10 04:20:18 +08:00
|
|
|
SDOperand BotNotZero = DAG.getSetCC(TLI.getSetCCResultTy(), LTZ, BitsC,
|
|
|
|
ISD::SETNE);
|
2005-05-13 03:05:01 +08:00
|
|
|
SDOperand HiPart = DAG.getNode(ISD::CTTZ, NVT, Hi);
|
|
|
|
HiPart = DAG.getNode(ISD::ADD, NVT, HiPart, BitsC);
|
|
|
|
|
|
|
|
Lo = DAG.getNode(ISD::SELECT, NVT, BotNotZero, LTZ, HiPart);
|
|
|
|
Hi = DAG.getConstant(0, NVT);
|
|
|
|
break;
|
|
|
|
}
|
2005-05-11 12:51:16 +08:00
|
|
|
|
2005-01-07 15:47:09 +08:00
|
|
|
case ISD::LOAD: {
|
|
|
|
SDOperand Ch = LegalizeOp(Node->getOperand(0)); // Legalize the chain.
|
|
|
|
SDOperand Ptr = LegalizeOp(Node->getOperand(1)); // Legalize the pointer.
|
2005-04-28 04:10:01 +08:00
|
|
|
Lo = DAG.getLoad(NVT, Ch, Ptr, Node->getOperand(2));
|
2005-01-07 15:47:09 +08:00
|
|
|
|
|
|
|
// Increment the pointer to the other half.
|
2005-01-10 03:43:23 +08:00
|
|
|
unsigned IncrementSize = MVT::getSizeInBits(Lo.getValueType())/8;
|
2005-01-07 15:47:09 +08:00
|
|
|
Ptr = DAG.getNode(ISD::ADD, Ptr.getValueType(), Ptr,
|
|
|
|
getIntPtrConstant(IncrementSize));
|
2005-07-27 14:12:32 +08:00
|
|
|
//Is this safe? declaring that the two parts of the split load
|
2005-04-28 04:10:01 +08:00
|
|
|
//are from the same instruction?
|
|
|
|
Hi = DAG.getLoad(NVT, Ch, Ptr, Node->getOperand(2));
|
2005-01-20 02:02:17 +08:00
|
|
|
|
|
|
|
// Build a factor node to remember that this load is independent of the
|
|
|
|
// other one.
|
|
|
|
SDOperand TF = DAG.getNode(ISD::TokenFactor, MVT::Other, Lo.getValue(1),
|
|
|
|
Hi.getValue(1));
|
2005-04-22 06:36:52 +08:00
|
|
|
|
2005-01-07 15:47:09 +08:00
|
|
|
// Remember that we legalized the chain.
|
2005-01-20 02:02:17 +08:00
|
|
|
AddLegalizedOperand(Op.getValue(1), TF);
|
2005-01-07 15:47:09 +08:00
|
|
|
if (!TLI.isLittleEndian())
|
|
|
|
std::swap(Lo, Hi);
|
|
|
|
break;
|
|
|
|
}
|
Check in code to scalarize arbitrarily wide packed types for some simple
vector operations (load, add, sub, mul).
This allows us to codegen:
void %foo(<4 x float> * %a) {
entry:
%tmp1 = load <4 x float> * %a;
%tmp2 = add <4 x float> %tmp1, %tmp1
store <4 x float> %tmp2, <4 x float> *%a
ret void
}
on ppc as:
_foo:
lfs f0, 12(r3)
lfs f1, 8(r3)
lfs f2, 4(r3)
lfs f3, 0(r3)
fadds f0, f0, f0
fadds f1, f1, f1
fadds f2, f2, f2
fadds f3, f3, f3
stfs f0, 12(r3)
stfs f1, 8(r3)
stfs f2, 4(r3)
stfs f3, 0(r3)
blr
llvm-svn: 24484
2005-11-23 02:16:00 +08:00
|
|
|
case ISD::VLOAD: {
|
|
|
|
SDOperand Ch = LegalizeOp(Node->getOperand(0)); // Legalize the chain.
|
|
|
|
SDOperand Ptr = LegalizeOp(Node->getOperand(1)); // Legalize the pointer.
|
|
|
|
unsigned NumElements =cast<ConstantSDNode>(Node->getOperand(2))->getValue();
|
|
|
|
MVT::ValueType EVT = cast<VTSDNode>(Node->getOperand(3))->getVT();
|
|
|
|
|
|
|
|
// If we only have two elements, turn into a pair of scalar loads.
|
|
|
|
// FIXME: handle case where a vector of two elements is fine, such as
|
|
|
|
// 2 x double on SSE2.
|
|
|
|
if (NumElements == 2) {
|
|
|
|
Lo = DAG.getLoad(EVT, Ch, Ptr, Node->getOperand(4));
|
|
|
|
// Increment the pointer to the other half.
|
|
|
|
unsigned IncrementSize = MVT::getSizeInBits(EVT)/8;
|
|
|
|
Ptr = DAG.getNode(ISD::ADD, Ptr.getValueType(), Ptr,
|
|
|
|
getIntPtrConstant(IncrementSize));
|
|
|
|
//Is this safe? declaring that the two parts of the split load
|
|
|
|
//are from the same instruction?
|
|
|
|
Hi = DAG.getLoad(EVT, Ch, Ptr, Node->getOperand(4));
|
|
|
|
} else {
|
|
|
|
NumElements /= 2; // Split the vector in half
|
|
|
|
Lo = DAG.getVecLoad(NumElements, EVT, Ch, Ptr, Node->getOperand(4));
|
|
|
|
unsigned IncrementSize = NumElements * MVT::getSizeInBits(EVT)/8;
|
|
|
|
Ptr = DAG.getNode(ISD::ADD, Ptr.getValueType(), Ptr,
|
|
|
|
getIntPtrConstant(IncrementSize));
|
|
|
|
//Is this safe? declaring that the two parts of the split load
|
|
|
|
//are from the same instruction?
|
|
|
|
Hi = DAG.getVecLoad(NumElements, EVT, Ch, Ptr, Node->getOperand(4));
|
|
|
|
}
|
|
|
|
|
|
|
|
// Build a factor node to remember that this load is independent of the
|
|
|
|
// other one.
|
|
|
|
SDOperand TF = DAG.getNode(ISD::TokenFactor, MVT::Other, Lo.getValue(1),
|
|
|
|
Hi.getValue(1));
|
|
|
|
|
|
|
|
// Remember that we legalized the chain.
|
|
|
|
AddLegalizedOperand(Op.getValue(1), TF);
|
|
|
|
if (!TLI.isLittleEndian())
|
|
|
|
std::swap(Lo, Hi);
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
case ISD::VADD:
|
|
|
|
case ISD::VSUB:
|
|
|
|
case ISD::VMUL: {
|
|
|
|
unsigned NumElements =cast<ConstantSDNode>(Node->getOperand(2))->getValue();
|
|
|
|
MVT::ValueType EVT = cast<VTSDNode>(Node->getOperand(3))->getVT();
|
|
|
|
SDOperand LL, LH, RL, RH;
|
|
|
|
|
|
|
|
ExpandOp(Node->getOperand(0), LL, LH);
|
|
|
|
ExpandOp(Node->getOperand(1), RL, RH);
|
|
|
|
|
|
|
|
// If we only have two elements, turn into a pair of scalar loads.
|
|
|
|
// FIXME: handle case where a vector of two elements is fine, such as
|
|
|
|
// 2 x double on SSE2.
|
|
|
|
if (NumElements == 2) {
|
|
|
|
unsigned Opc = getScalarizedOpcode(Node->getOpcode(), EVT);
|
|
|
|
Lo = DAG.getNode(Opc, EVT, LL, RL);
|
|
|
|
Hi = DAG.getNode(Opc, EVT, LH, RH);
|
|
|
|
} else {
|
|
|
|
Lo = DAG.getNode(Node->getOpcode(), MVT::Vector, LL, RL, LL.getOperand(2),
|
|
|
|
LL.getOperand(3));
|
|
|
|
Hi = DAG.getNode(Node->getOpcode(), MVT::Vector, LH, RH, LH.getOperand(2),
|
|
|
|
LH.getOperand(3));
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
}
|
2005-05-14 02:43:43 +08:00
|
|
|
case ISD::TAILCALL:
|
2005-01-07 15:47:09 +08:00
|
|
|
case ISD::CALL: {
|
|
|
|
SDOperand Chain = LegalizeOp(Node->getOperand(0)); // Legalize the chain.
|
|
|
|
SDOperand Callee = LegalizeOp(Node->getOperand(1)); // Legalize the callee.
|
|
|
|
|
2005-01-20 04:24:35 +08:00
|
|
|
bool Changed = false;
|
|
|
|
std::vector<SDOperand> Ops;
|
|
|
|
for (unsigned i = 2, e = Node->getNumOperands(); i != e; ++i) {
|
|
|
|
Ops.push_back(LegalizeOp(Node->getOperand(i)));
|
|
|
|
Changed |= Ops.back() != Node->getOperand(i);
|
|
|
|
}
|
|
|
|
|
2005-01-07 15:47:09 +08:00
|
|
|
assert(Node->getNumValues() == 2 && Op.ResNo == 0 &&
|
|
|
|
"Can only expand a call once so far, not i64 -> i16!");
|
|
|
|
|
|
|
|
std::vector<MVT::ValueType> RetTyVTs;
|
|
|
|
RetTyVTs.reserve(3);
|
|
|
|
RetTyVTs.push_back(NVT);
|
|
|
|
RetTyVTs.push_back(NVT);
|
|
|
|
RetTyVTs.push_back(MVT::Other);
|
2005-05-14 02:43:43 +08:00
|
|
|
SDNode *NC = DAG.getCall(RetTyVTs, Chain, Callee, Ops,
|
|
|
|
Node->getOpcode() == ISD::TAILCALL);
|
2005-01-07 15:47:09 +08:00
|
|
|
Lo = SDOperand(NC, 0);
|
|
|
|
Hi = SDOperand(NC, 1);
|
|
|
|
|
|
|
|
// Insert the new chain mapping.
|
2005-01-09 04:35:13 +08:00
|
|
|
AddLegalizedOperand(Op.getValue(1), Hi.getValue(2));
|
2005-01-07 15:47:09 +08:00
|
|
|
break;
|
|
|
|
}
|
|
|
|
case ISD::AND:
|
|
|
|
case ISD::OR:
|
|
|
|
case ISD::XOR: { // Simple logical operators -> two trivial pieces.
|
|
|
|
SDOperand LL, LH, RL, RH;
|
|
|
|
ExpandOp(Node->getOperand(0), LL, LH);
|
|
|
|
ExpandOp(Node->getOperand(1), RL, RH);
|
|
|
|
Lo = DAG.getNode(Node->getOpcode(), NVT, LL, RL);
|
|
|
|
Hi = DAG.getNode(Node->getOpcode(), NVT, LH, RH);
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
case ISD::SELECT: {
|
|
|
|
SDOperand C, LL, LH, RL, RH;
|
2005-01-19 03:27:06 +08:00
|
|
|
|
|
|
|
switch (getTypeAction(Node->getOperand(0).getValueType())) {
|
|
|
|
case Expand: assert(0 && "It's impossible to expand bools");
|
|
|
|
case Legal:
|
|
|
|
C = LegalizeOp(Node->getOperand(0)); // Legalize the condition.
|
|
|
|
break;
|
|
|
|
case Promote:
|
|
|
|
C = PromoteOp(Node->getOperand(0)); // Promote the condition.
|
|
|
|
break;
|
|
|
|
}
|
2005-01-07 15:47:09 +08:00
|
|
|
ExpandOp(Node->getOperand(1), LL, LH);
|
|
|
|
ExpandOp(Node->getOperand(2), RL, RH);
|
|
|
|
Lo = DAG.getNode(ISD::SELECT, NVT, C, LL, RL);
|
|
|
|
Hi = DAG.getNode(ISD::SELECT, NVT, C, LH, RH);
|
|
|
|
break;
|
|
|
|
}
|
2005-08-11 04:51:12 +08:00
|
|
|
case ISD::SELECT_CC: {
|
|
|
|
SDOperand TL, TH, FL, FH;
|
|
|
|
ExpandOp(Node->getOperand(2), TL, TH);
|
|
|
|
ExpandOp(Node->getOperand(3), FL, FH);
|
|
|
|
Lo = DAG.getNode(ISD::SELECT_CC, NVT, Node->getOperand(0),
|
|
|
|
Node->getOperand(1), TL, FL, Node->getOperand(4));
|
|
|
|
Hi = DAG.getNode(ISD::SELECT_CC, NVT, Node->getOperand(0),
|
|
|
|
Node->getOperand(1), TH, FH, Node->getOperand(4));
|
2005-08-11 09:12:20 +08:00
|
|
|
Lo = LegalizeOp(Lo);
|
|
|
|
Hi = LegalizeOp(Hi);
|
2005-08-11 04:51:12 +08:00
|
|
|
break;
|
|
|
|
}
|
2005-10-14 01:15:37 +08:00
|
|
|
case ISD::SEXTLOAD: {
|
|
|
|
SDOperand Chain = LegalizeOp(Node->getOperand(0));
|
|
|
|
SDOperand Ptr = LegalizeOp(Node->getOperand(1));
|
|
|
|
MVT::ValueType EVT = cast<VTSDNode>(Node->getOperand(3))->getVT();
|
|
|
|
|
|
|
|
if (EVT == NVT)
|
|
|
|
Lo = DAG.getLoad(NVT, Chain, Ptr, Node->getOperand(2));
|
|
|
|
else
|
|
|
|
Lo = DAG.getExtLoad(ISD::SEXTLOAD, NVT, Chain, Ptr, Node->getOperand(2),
|
|
|
|
EVT);
|
2005-10-14 05:44:47 +08:00
|
|
|
|
|
|
|
// Remember that we legalized the chain.
|
|
|
|
AddLegalizedOperand(SDOperand(Node, 1), Lo.getValue(1));
|
|
|
|
|
2005-10-14 01:15:37 +08:00
|
|
|
// The high part is obtained by SRA'ing all but one of the bits of the lo
|
|
|
|
// part.
|
|
|
|
unsigned LoSize = MVT::getSizeInBits(Lo.getValueType());
|
|
|
|
Hi = DAG.getNode(ISD::SRA, NVT, Lo, DAG.getConstant(LoSize-1,
|
|
|
|
TLI.getShiftAmountTy()));
|
|
|
|
Lo = LegalizeOp(Lo);
|
|
|
|
Hi = LegalizeOp(Hi);
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
case ISD::ZEXTLOAD: {
|
|
|
|
SDOperand Chain = LegalizeOp(Node->getOperand(0));
|
|
|
|
SDOperand Ptr = LegalizeOp(Node->getOperand(1));
|
|
|
|
MVT::ValueType EVT = cast<VTSDNode>(Node->getOperand(3))->getVT();
|
|
|
|
|
|
|
|
if (EVT == NVT)
|
|
|
|
Lo = DAG.getLoad(NVT, Chain, Ptr, Node->getOperand(2));
|
|
|
|
else
|
|
|
|
Lo = DAG.getExtLoad(ISD::ZEXTLOAD, NVT, Chain, Ptr, Node->getOperand(2),
|
|
|
|
EVT);
|
2005-10-14 05:44:47 +08:00
|
|
|
|
|
|
|
// Remember that we legalized the chain.
|
|
|
|
AddLegalizedOperand(SDOperand(Node, 1), Lo.getValue(1));
|
|
|
|
|
2005-10-14 01:15:37 +08:00
|
|
|
// The high part is just a zero.
|
2005-10-14 05:44:47 +08:00
|
|
|
Hi = LegalizeOp(DAG.getConstant(0, NVT));
|
|
|
|
Lo = LegalizeOp(Lo);
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
case ISD::EXTLOAD: {
|
|
|
|
SDOperand Chain = LegalizeOp(Node->getOperand(0));
|
|
|
|
SDOperand Ptr = LegalizeOp(Node->getOperand(1));
|
|
|
|
MVT::ValueType EVT = cast<VTSDNode>(Node->getOperand(3))->getVT();
|
|
|
|
|
|
|
|
if (EVT == NVT)
|
|
|
|
Lo = DAG.getLoad(NVT, Chain, Ptr, Node->getOperand(2));
|
|
|
|
else
|
|
|
|
Lo = DAG.getExtLoad(ISD::EXTLOAD, NVT, Chain, Ptr, Node->getOperand(2),
|
|
|
|
EVT);
|
|
|
|
|
|
|
|
// Remember that we legalized the chain.
|
|
|
|
AddLegalizedOperand(SDOperand(Node, 1), Lo.getValue(1));
|
|
|
|
|
|
|
|
// The high part is undefined.
|
|
|
|
Hi = LegalizeOp(DAG.getNode(ISD::UNDEF, NVT));
|
2005-10-14 01:15:37 +08:00
|
|
|
Lo = LegalizeOp(Lo);
|
|
|
|
break;
|
|
|
|
}
|
2005-09-02 08:18:10 +08:00
|
|
|
case ISD::ANY_EXTEND: {
|
|
|
|
SDOperand In;
|
|
|
|
switch (getTypeAction(Node->getOperand(0).getValueType())) {
|
|
|
|
case Expand: assert(0 && "expand-expand not implemented yet!");
|
|
|
|
case Legal: In = LegalizeOp(Node->getOperand(0)); break;
|
|
|
|
case Promote:
|
|
|
|
In = PromoteOp(Node->getOperand(0));
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
|
|
|
// The low part is any extension of the input (which degenerates to a copy).
|
|
|
|
Lo = DAG.getNode(ISD::ANY_EXTEND, NVT, In);
|
|
|
|
// The high part is undefined.
|
|
|
|
Hi = DAG.getNode(ISD::UNDEF, NVT);
|
|
|
|
break;
|
|
|
|
}
|
2005-01-07 15:47:09 +08:00
|
|
|
case ISD::SIGN_EXTEND: {
|
2005-04-04 07:41:52 +08:00
|
|
|
SDOperand In;
|
|
|
|
switch (getTypeAction(Node->getOperand(0).getValueType())) {
|
|
|
|
case Expand: assert(0 && "expand-expand not implemented yet!");
|
|
|
|
case Legal: In = LegalizeOp(Node->getOperand(0)); break;
|
|
|
|
case Promote:
|
|
|
|
In = PromoteOp(Node->getOperand(0));
|
|
|
|
// Emit the appropriate sign_extend_inreg to get the value we want.
|
|
|
|
In = DAG.getNode(ISD::SIGN_EXTEND_INREG, In.getValueType(), In,
|
2005-07-10 08:07:11 +08:00
|
|
|
DAG.getValueType(Node->getOperand(0).getValueType()));
|
2005-04-04 07:41:52 +08:00
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
2005-01-07 15:47:09 +08:00
|
|
|
// The low part is just a sign extension of the input (which degenerates to
|
|
|
|
// a copy).
|
2005-04-04 07:41:52 +08:00
|
|
|
Lo = DAG.getNode(ISD::SIGN_EXTEND, NVT, In);
|
2005-04-22 06:36:52 +08:00
|
|
|
|
2005-01-07 15:47:09 +08:00
|
|
|
// The high part is obtained by SRA'ing all but one of the bits of the lo
|
|
|
|
// part.
|
2005-01-13 02:19:52 +08:00
|
|
|
unsigned LoSize = MVT::getSizeInBits(Lo.getValueType());
|
2005-01-22 08:31:52 +08:00
|
|
|
Hi = DAG.getNode(ISD::SRA, NVT, Lo, DAG.getConstant(LoSize-1,
|
|
|
|
TLI.getShiftAmountTy()));
|
2005-01-07 15:47:09 +08:00
|
|
|
break;
|
|
|
|
}
|
2005-04-04 07:41:52 +08:00
|
|
|
case ISD::ZERO_EXTEND: {
|
|
|
|
SDOperand In;
|
|
|
|
switch (getTypeAction(Node->getOperand(0).getValueType())) {
|
|
|
|
case Expand: assert(0 && "expand-expand not implemented yet!");
|
|
|
|
case Legal: In = LegalizeOp(Node->getOperand(0)); break;
|
|
|
|
case Promote:
|
|
|
|
In = PromoteOp(Node->getOperand(0));
|
|
|
|
// Emit the appropriate zero_extend_inreg to get the value we want.
|
2005-04-13 10:38:47 +08:00
|
|
|
In = DAG.getZeroExtendInReg(In, Node->getOperand(0).getValueType());
|
2005-04-04 07:41:52 +08:00
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
2005-01-07 15:47:09 +08:00
|
|
|
// The low part is just a zero extension of the input (which degenerates to
|
|
|
|
// a copy).
|
2005-04-10 09:13:15 +08:00
|
|
|
Lo = DAG.getNode(ISD::ZERO_EXTEND, NVT, In);
|
2005-04-22 06:36:52 +08:00
|
|
|
|
2005-01-07 15:47:09 +08:00
|
|
|
// The high part is just a zero.
|
|
|
|
Hi = DAG.getConstant(0, NVT);
|
|
|
|
break;
|
2005-04-04 07:41:52 +08:00
|
|
|
}
|
2005-12-23 08:16:34 +08:00
|
|
|
|
|
|
|
case ISD::BIT_CONVERT: {
|
|
|
|
SDOperand Tmp = ExpandBIT_CONVERT(Node->getValueType(0),
|
|
|
|
Node->getOperand(0));
|
|
|
|
ExpandOp(Tmp, Lo, Hi);
|
|
|
|
break;
|
|
|
|
}
|
2005-11-21 05:32:07 +08:00
|
|
|
|
2005-11-21 06:56:56 +08:00
|
|
|
case ISD::READCYCLECOUNTER: {
|
|
|
|
assert(TLI.getOperationAction(ISD::READCYCLECOUNTER, VT) ==
|
|
|
|
TargetLowering::Custom &&
|
|
|
|
"Must custom expand ReadCycleCounter");
|
|
|
|
SDOperand T = TLI.LowerOperation(Op, DAG);
|
|
|
|
assert(T.Val && "Node must be custom expanded!");
|
|
|
|
Lo = LegalizeOp(T.getValue(0));
|
|
|
|
Hi = LegalizeOp(T.getValue(1));
|
|
|
|
AddLegalizedOperand(SDOperand(Node, 1), // Remember we legalized the chain.
|
|
|
|
LegalizeOp(T.getValue(2)));
|
2005-11-21 05:32:07 +08:00
|
|
|
break;
|
2005-11-21 06:56:56 +08:00
|
|
|
}
|
2005-11-21 05:32:07 +08:00
|
|
|
|
2005-01-09 03:27:05 +08:00
|
|
|
// These operators cannot be expanded directly, emit them as calls to
|
|
|
|
// library functions.
|
|
|
|
case ISD::FP_TO_SINT:
|
2005-07-29 08:33:32 +08:00
|
|
|
if (TLI.getOperationAction(ISD::FP_TO_SINT, VT) == TargetLowering::Custom) {
|
2005-07-30 09:40:57 +08:00
|
|
|
SDOperand Op;
|
|
|
|
switch (getTypeAction(Node->getOperand(0).getValueType())) {
|
|
|
|
case Expand: assert(0 && "cannot expand FP!");
|
|
|
|
case Legal: Op = LegalizeOp(Node->getOperand(0)); break;
|
|
|
|
case Promote: Op = PromoteOp(Node->getOperand(0)); break;
|
|
|
|
}
|
2005-07-31 02:33:25 +08:00
|
|
|
|
2005-07-30 09:40:57 +08:00
|
|
|
Op = TLI.LowerOperation(DAG.getNode(ISD::FP_TO_SINT, VT, Op), DAG);
|
|
|
|
|
2005-07-29 08:33:32 +08:00
|
|
|
// Now that the custom expander is done, expand the result, which is still
|
|
|
|
// VT.
|
2005-08-26 08:14:16 +08:00
|
|
|
if (Op.Val) {
|
|
|
|
ExpandOp(Op, Lo, Hi);
|
|
|
|
break;
|
|
|
|
}
|
2005-07-29 08:33:32 +08:00
|
|
|
}
|
2005-07-31 02:33:25 +08:00
|
|
|
|
2005-01-09 03:27:05 +08:00
|
|
|
if (Node->getOperand(0).getValueType() == MVT::f32)
|
2005-01-21 14:05:23 +08:00
|
|
|
Lo = ExpandLibCall("__fixsfdi", Node, Hi);
|
2005-01-09 03:27:05 +08:00
|
|
|
else
|
2005-01-21 14:05:23 +08:00
|
|
|
Lo = ExpandLibCall("__fixdfdi", Node, Hi);
|
2005-01-09 03:27:05 +08:00
|
|
|
break;
|
2005-07-31 02:33:25 +08:00
|
|
|
|
2005-01-09 03:27:05 +08:00
|
|
|
case ISD::FP_TO_UINT:
|
2005-07-29 08:33:32 +08:00
|
|
|
if (TLI.getOperationAction(ISD::FP_TO_UINT, VT) == TargetLowering::Custom) {
|
|
|
|
SDOperand Op = DAG.getNode(ISD::FP_TO_UINT, VT,
|
|
|
|
LegalizeOp(Node->getOperand(0)));
|
|
|
|
// Now that the custom expander is done, expand the result, which is still
|
|
|
|
// VT.
|
2005-08-26 08:14:16 +08:00
|
|
|
Op = TLI.LowerOperation(Op, DAG);
|
|
|
|
if (Op.Val) {
|
|
|
|
ExpandOp(Op, Lo, Hi);
|
|
|
|
break;
|
|
|
|
}
|
2005-07-29 08:33:32 +08:00
|
|
|
}
|
2005-07-31 02:33:25 +08:00
|
|
|
|
2005-01-09 03:27:05 +08:00
|
|
|
if (Node->getOperand(0).getValueType() == MVT::f32)
|
2005-01-21 14:05:23 +08:00
|
|
|
Lo = ExpandLibCall("__fixunssfdi", Node, Hi);
|
2005-01-09 03:27:05 +08:00
|
|
|
else
|
2005-01-21 14:05:23 +08:00
|
|
|
Lo = ExpandLibCall("__fixunsdfdi", Node, Hi);
|
2005-01-09 03:27:05 +08:00
|
|
|
break;
|
|
|
|
|
2006-01-10 02:31:59 +08:00
|
|
|
case ISD::SHL: {
|
2005-09-01 03:01:53 +08:00
|
|
|
// If the target wants custom lowering, do so.
|
2006-01-21 12:27:00 +08:00
|
|
|
SDOperand ShiftAmt = LegalizeOp(Node->getOperand(1));
|
2005-09-01 03:01:53 +08:00
|
|
|
if (TLI.getOperationAction(ISD::SHL, VT) == TargetLowering::Custom) {
|
|
|
|
SDOperand Op = DAG.getNode(ISD::SHL, VT, Node->getOperand(0),
|
2006-01-21 12:27:00 +08:00
|
|
|
ShiftAmt);
|
2005-09-01 03:01:53 +08:00
|
|
|
Op = TLI.LowerOperation(Op, DAG);
|
|
|
|
if (Op.Val) {
|
|
|
|
// Now that the custom expander is done, expand the result, which is
|
|
|
|
// still VT.
|
|
|
|
ExpandOp(Op, Lo, Hi);
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2005-01-19 12:19:40 +08:00
|
|
|
// If we can emit an efficient shift operation, do so now.
|
2006-01-21 12:27:00 +08:00
|
|
|
if (ExpandShift(ISD::SHL, Node->getOperand(0), ShiftAmt, Lo, Hi))
|
2005-01-19 12:19:40 +08:00
|
|
|
break;
|
2005-04-02 11:38:53 +08:00
|
|
|
|
|
|
|
// If this target supports SHL_PARTS, use it.
|
2006-01-10 02:31:59 +08:00
|
|
|
TargetLowering::LegalizeAction Action =
|
|
|
|
TLI.getOperationAction(ISD::SHL_PARTS, NVT);
|
|
|
|
if ((Action == TargetLowering::Legal && TLI.isTypeLegal(NVT)) ||
|
|
|
|
Action == TargetLowering::Custom) {
|
2006-01-21 12:27:00 +08:00
|
|
|
ExpandShiftParts(ISD::SHL_PARTS, Node->getOperand(0), ShiftAmt, Lo, Hi);
|
2005-04-02 11:38:53 +08:00
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
2005-01-19 12:19:40 +08:00
|
|
|
// Otherwise, emit a libcall.
|
2005-01-21 14:05:23 +08:00
|
|
|
Lo = ExpandLibCall("__ashldi3", Node, Hi);
|
2005-01-19 12:19:40 +08:00
|
|
|
break;
|
2006-01-10 02:31:59 +08:00
|
|
|
}
|
2005-01-19 12:19:40 +08:00
|
|
|
|
2006-01-10 02:31:59 +08:00
|
|
|
case ISD::SRA: {
|
2005-09-01 03:01:53 +08:00
|
|
|
// If the target wants custom lowering, do so.
|
2006-01-21 12:27:00 +08:00
|
|
|
SDOperand ShiftAmt = LegalizeOp(Node->getOperand(1));
|
2005-09-01 03:01:53 +08:00
|
|
|
if (TLI.getOperationAction(ISD::SRA, VT) == TargetLowering::Custom) {
|
|
|
|
SDOperand Op = DAG.getNode(ISD::SRA, VT, Node->getOperand(0),
|
2006-01-21 12:27:00 +08:00
|
|
|
ShiftAmt);
|
2005-09-01 03:01:53 +08:00
|
|
|
Op = TLI.LowerOperation(Op, DAG);
|
|
|
|
if (Op.Val) {
|
|
|
|
// Now that the custom expander is done, expand the result, which is
|
|
|
|
// still VT.
|
|
|
|
ExpandOp(Op, Lo, Hi);
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2005-01-19 12:19:40 +08:00
|
|
|
// If we can emit an efficient shift operation, do so now.
|
2006-01-21 12:27:00 +08:00
|
|
|
if (ExpandShift(ISD::SRA, Node->getOperand(0), ShiftAmt, Lo, Hi))
|
2005-01-19 12:19:40 +08:00
|
|
|
break;
|
2005-04-02 11:38:53 +08:00
|
|
|
|
|
|
|
// If this target supports SRA_PARTS, use it.
|
2006-01-10 02:31:59 +08:00
|
|
|
TargetLowering::LegalizeAction Action =
|
|
|
|
TLI.getOperationAction(ISD::SRA_PARTS, NVT);
|
|
|
|
if ((Action == TargetLowering::Legal && TLI.isTypeLegal(NVT)) ||
|
|
|
|
Action == TargetLowering::Custom) {
|
2006-01-21 12:27:00 +08:00
|
|
|
ExpandShiftParts(ISD::SRA_PARTS, Node->getOperand(0), ShiftAmt, Lo, Hi);
|
2005-04-02 11:38:53 +08:00
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
2005-01-19 12:19:40 +08:00
|
|
|
// Otherwise, emit a libcall.
|
2005-01-21 14:05:23 +08:00
|
|
|
Lo = ExpandLibCall("__ashrdi3", Node, Hi);
|
2005-01-19 12:19:40 +08:00
|
|
|
break;
|
2006-01-10 02:31:59 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
case ISD::SRL: {
|
2005-09-01 03:01:53 +08:00
|
|
|
// If the target wants custom lowering, do so.
|
2006-01-21 12:27:00 +08:00
|
|
|
SDOperand ShiftAmt = LegalizeOp(Node->getOperand(1));
|
2005-09-01 03:01:53 +08:00
|
|
|
if (TLI.getOperationAction(ISD::SRL, VT) == TargetLowering::Custom) {
|
|
|
|
SDOperand Op = DAG.getNode(ISD::SRL, VT, Node->getOperand(0),
|
2006-01-21 12:27:00 +08:00
|
|
|
ShiftAmt);
|
2005-09-01 03:01:53 +08:00
|
|
|
Op = TLI.LowerOperation(Op, DAG);
|
|
|
|
if (Op.Val) {
|
|
|
|
// Now that the custom expander is done, expand the result, which is
|
|
|
|
// still VT.
|
|
|
|
ExpandOp(Op, Lo, Hi);
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2005-01-19 12:19:40 +08:00
|
|
|
// If we can emit an efficient shift operation, do so now.
|
2006-01-21 12:27:00 +08:00
|
|
|
if (ExpandShift(ISD::SRL, Node->getOperand(0), ShiftAmt, Lo, Hi))
|
2005-01-19 12:19:40 +08:00
|
|
|
break;
|
2005-04-02 11:38:53 +08:00
|
|
|
|
|
|
|
// If this target supports SRL_PARTS, use it.
|
2006-01-10 02:31:59 +08:00
|
|
|
TargetLowering::LegalizeAction Action =
|
|
|
|
TLI.getOperationAction(ISD::SRL_PARTS, NVT);
|
|
|
|
if ((Action == TargetLowering::Legal && TLI.isTypeLegal(NVT)) ||
|
|
|
|
Action == TargetLowering::Custom) {
|
2006-01-21 12:27:00 +08:00
|
|
|
ExpandShiftParts(ISD::SRL_PARTS, Node->getOperand(0), ShiftAmt, Lo, Hi);
|
2005-04-02 11:38:53 +08:00
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
2005-01-19 12:19:40 +08:00
|
|
|
// Otherwise, emit a libcall.
|
2005-01-21 14:05:23 +08:00
|
|
|
Lo = ExpandLibCall("__lshrdi3", Node, Hi);
|
2005-01-19 12:19:40 +08:00
|
|
|
break;
|
2006-01-10 02:31:59 +08:00
|
|
|
}
|
2005-01-19 12:19:40 +08:00
|
|
|
|
2005-04-22 06:36:52 +08:00
|
|
|
case ISD::ADD:
|
2005-04-02 11:38:53 +08:00
|
|
|
ExpandByParts(ISD::ADD_PARTS, Node->getOperand(0), Node->getOperand(1),
|
|
|
|
Lo, Hi);
|
2005-01-21 02:52:28 +08:00
|
|
|
break;
|
|
|
|
case ISD::SUB:
|
2005-04-02 11:38:53 +08:00
|
|
|
ExpandByParts(ISD::SUB_PARTS, Node->getOperand(0), Node->getOperand(1),
|
|
|
|
Lo, Hi);
|
2005-01-21 02:52:28 +08:00
|
|
|
break;
|
2005-04-11 11:01:51 +08:00
|
|
|
case ISD::MUL: {
|
2005-08-25 00:35:28 +08:00
|
|
|
if (TLI.isOperationLegal(ISD::MULHU, NVT)) {
|
2005-04-11 11:01:51 +08:00
|
|
|
SDOperand LL, LH, RL, RH;
|
|
|
|
ExpandOp(Node->getOperand(0), LL, LH);
|
|
|
|
ExpandOp(Node->getOperand(1), RL, RH);
|
Add support for AssertSext and AssertZext, folding other extensions with
them. This allows for elminination of redundant extends in the entry
blocks of functions on PowerPC.
Add support for i32 x i32 -> i64 multiplies, by recognizing when the inputs
to ISD::MUL in ExpandOp are actually just extended i32 values and not real
i64 values. this allows us to codegen
int mulhs(int a, int b) { return ((long long)a * b) >> 32; }
as:
_mulhs:
mulhw r3, r4, r3
blr
instead of:
_mulhs:
mulhwu r2, r4, r3
srawi r5, r3, 31
mullw r5, r4, r5
add r2, r2, r5
srawi r4, r4, 31
mullw r3, r4, r3
add r3, r2, r3
blr
with a similar improvement on x86.
llvm-svn: 23147
2005-08-30 10:44:00 +08:00
|
|
|
unsigned SH = MVT::getSizeInBits(RH.getValueType())-1;
|
|
|
|
// MULHS implicitly sign extends its inputs. Check to see if ExpandOp
|
|
|
|
// extended the sign bit of the low half through the upper half, and if so
|
|
|
|
// emit a MULHS instead of the alternate sequence that is valid for any
|
|
|
|
// i64 x i64 multiply.
|
|
|
|
if (TLI.isOperationLegal(ISD::MULHS, NVT) &&
|
|
|
|
// is RH an extension of the sign bit of RL?
|
|
|
|
RH.getOpcode() == ISD::SRA && RH.getOperand(0) == RL &&
|
|
|
|
RH.getOperand(1).getOpcode() == ISD::Constant &&
|
|
|
|
cast<ConstantSDNode>(RH.getOperand(1))->getValue() == SH &&
|
|
|
|
// is LH an extension of the sign bit of LL?
|
|
|
|
LH.getOpcode() == ISD::SRA && LH.getOperand(0) == LL &&
|
|
|
|
LH.getOperand(1).getOpcode() == ISD::Constant &&
|
|
|
|
cast<ConstantSDNode>(LH.getOperand(1))->getValue() == SH) {
|
|
|
|
Hi = DAG.getNode(ISD::MULHS, NVT, LL, RL);
|
|
|
|
} else {
|
|
|
|
Hi = DAG.getNode(ISD::MULHU, NVT, LL, RL);
|
|
|
|
RH = DAG.getNode(ISD::MUL, NVT, LL, RH);
|
|
|
|
LH = DAG.getNode(ISD::MUL, NVT, LH, RL);
|
|
|
|
Hi = DAG.getNode(ISD::ADD, NVT, Hi, RH);
|
|
|
|
Hi = DAG.getNode(ISD::ADD, NVT, Hi, LH);
|
|
|
|
}
|
2005-04-11 11:01:51 +08:00
|
|
|
Lo = DAG.getNode(ISD::MUL, NVT, LL, RL);
|
|
|
|
} else {
|
|
|
|
Lo = ExpandLibCall("__muldi3" , Node, Hi); break;
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
}
|
2005-01-21 14:05:23 +08:00
|
|
|
case ISD::SDIV: Lo = ExpandLibCall("__divdi3" , Node, Hi); break;
|
|
|
|
case ISD::UDIV: Lo = ExpandLibCall("__udivdi3", Node, Hi); break;
|
|
|
|
case ISD::SREM: Lo = ExpandLibCall("__moddi3" , Node, Hi); break;
|
|
|
|
case ISD::UREM: Lo = ExpandLibCall("__umoddi3", Node, Hi); break;
|
2005-01-07 15:47:09 +08:00
|
|
|
}
|
|
|
|
|
2005-12-22 02:02:52 +08:00
|
|
|
// Make sure the resultant values have been legalized themselves, unless this
|
|
|
|
// is a type that requires multi-step expansion.
|
|
|
|
if (getTypeAction(NVT) != Expand && NVT != MVT::isVoid) {
|
|
|
|
Lo = LegalizeOp(Lo);
|
|
|
|
Hi = LegalizeOp(Hi);
|
|
|
|
}
|
2006-01-10 02:31:59 +08:00
|
|
|
|
|
|
|
// Remember in a map if the values will be reused later.
|
|
|
|
bool isNew =
|
|
|
|
ExpandedNodes.insert(std::make_pair(Op, std::make_pair(Lo, Hi))).second;
|
|
|
|
assert(isNew && "Value already expanded?!?");
|
2005-01-07 15:47:09 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
// SelectionDAG::Legalize - This is the entry point for the file.
|
|
|
|
//
|
2005-01-23 12:42:50 +08:00
|
|
|
void SelectionDAG::Legalize() {
|
2005-01-07 15:47:09 +08:00
|
|
|
/// run - This is the main entry point to this class.
|
|
|
|
///
|
2005-01-23 12:42:50 +08:00
|
|
|
SelectionDAGLegalize(*this).Run();
|
2005-01-07 15:47:09 +08:00
|
|
|
}
|
|
|
|
|