@@ -899,6 +899,39 @@ void SelectionDAGLegalize::LegalizeLoadOps(SDNode *Node) {
899
899
}
900
900
}
901
901
902
+ static TargetLowering::LegalizeAction
903
+ getStrictFPOpcodeAction (const TargetLowering &TLI, unsigned Opcode, EVT VT) {
904
+ unsigned EqOpc;
905
+ switch (Opcode) {
906
+ default : llvm_unreachable (" Unexpected FP pseudo-opcode" );
907
+ case ISD::STRICT_FSQRT: EqOpc = ISD::FSQRT; break ;
908
+ case ISD::STRICT_FPOW: EqOpc = ISD::FPOW; break ;
909
+ case ISD::STRICT_FPOWI: EqOpc = ISD::FPOWI; break ;
910
+ case ISD::STRICT_FSIN: EqOpc = ISD::FSIN; break ;
911
+ case ISD::STRICT_FCOS: EqOpc = ISD::FCOS; break ;
912
+ case ISD::STRICT_FEXP: EqOpc = ISD::FEXP; break ;
913
+ case ISD::STRICT_FEXP2: EqOpc = ISD::FEXP2; break ;
914
+ case ISD::STRICT_FLOG: EqOpc = ISD::FLOG; break ;
915
+ case ISD::STRICT_FLOG10: EqOpc = ISD::FLOG10; break ;
916
+ case ISD::STRICT_FLOG2: EqOpc = ISD::FLOG2; break ;
917
+ case ISD::STRICT_FRINT: EqOpc = ISD::FRINT; break ;
918
+ case ISD::STRICT_FNEARBYINT: EqOpc = ISD::FNEARBYINT; break ;
919
+ }
920
+
921
+ auto Action = TLI.getOperationAction (EqOpc, VT);
922
+
923
+ // We don't currently handle Custom or Promote for strict FP pseudo-ops.
924
+ // For now, we just expand for those cases.
925
+ if (Action != TargetLowering::Legal)
926
+ Action = TargetLowering::Expand;
927
+
928
+ // ISD::FPOWI returns 'Legal' even though it should be expanded.
929
+ if (Opcode == ISD::STRICT_FPOWI && Action == TargetLowering::Legal)
930
+ Action = TargetLowering::Expand;
931
+
932
+ return Action;
933
+ }
934
+
902
935
// / Return a legal replacement for the given operation, with all legal operands.
903
936
void SelectionDAGLegalize::LegalizeOp (SDNode *Node) {
904
937
DEBUG (dbgs () << " \n Legalizing: " ; Node->dump (&DAG));
@@ -1043,6 +1076,25 @@ void SelectionDAGLegalize::LegalizeOp(SDNode *Node) {
1043
1076
return ;
1044
1077
}
1045
1078
break ;
1079
+ case ISD::STRICT_FSQRT:
1080
+ case ISD::STRICT_FPOW:
1081
+ case ISD::STRICT_FPOWI:
1082
+ case ISD::STRICT_FSIN:
1083
+ case ISD::STRICT_FCOS:
1084
+ case ISD::STRICT_FEXP:
1085
+ case ISD::STRICT_FEXP2:
1086
+ case ISD::STRICT_FLOG:
1087
+ case ISD::STRICT_FLOG10:
1088
+ case ISD::STRICT_FLOG2:
1089
+ case ISD::STRICT_FRINT:
1090
+ case ISD::STRICT_FNEARBYINT:
1091
+ // These pseudo-ops get legalized as if they were their non-strict
1092
+ // equivalent. For instance, if ISD::FSQRT is legal then ISD::STRICT_FSQRT
1093
+ // is also legal, but if ISD::FSQRT requires expansion then so does
1094
+ // ISD::STRICT_FSQRT.
1095
+ Action = getStrictFPOpcodeAction (TLI, Node->getOpcode (),
1096
+ Node->getValueType (0 ));
1097
+ break ;
1046
1098
1047
1099
default :
1048
1100
if (Node->getOpcode () >= ISD::BUILTIN_OP_END) {
@@ -2032,6 +2084,9 @@ SDValue SelectionDAGLegalize::ExpandFPLibCall(SDNode* Node,
2032
2084
RTLIB::Libcall Call_F80,
2033
2085
RTLIB::Libcall Call_F128,
2034
2086
RTLIB::Libcall Call_PPCF128) {
2087
+ if (Node->isStrictFPOpcode ())
2088
+ Node = DAG.mutateStrictFPToFP (Node);
2089
+
2035
2090
RTLIB::Libcall LC;
2036
2091
switch (Node->getSimpleValueType (0 ).SimpleTy ) {
2037
2092
default : llvm_unreachable (" Unexpected request for libcall!" );
@@ -3907,16 +3962,19 @@ void SelectionDAGLegalize::ConvertNodeToLibcall(SDNode *Node) {
3907
3962
RTLIB::FMAX_PPCF128));
3908
3963
break ;
3909
3964
case ISD::FSQRT:
3965
+ case ISD::STRICT_FSQRT:
3910
3966
Results.push_back (ExpandFPLibCall (Node, RTLIB::SQRT_F32, RTLIB::SQRT_F64,
3911
3967
RTLIB::SQRT_F80, RTLIB::SQRT_F128,
3912
3968
RTLIB::SQRT_PPCF128));
3913
3969
break ;
3914
3970
case ISD::FSIN:
3971
+ case ISD::STRICT_FSIN:
3915
3972
Results.push_back (ExpandFPLibCall (Node, RTLIB::SIN_F32, RTLIB::SIN_F64,
3916
3973
RTLIB::SIN_F80, RTLIB::SIN_F128,
3917
3974
RTLIB::SIN_PPCF128));
3918
3975
break ;
3919
3976
case ISD::FCOS:
3977
+ case ISD::STRICT_FCOS:
3920
3978
Results.push_back (ExpandFPLibCall (Node, RTLIB::COS_F32, RTLIB::COS_F64,
3921
3979
RTLIB::COS_F80, RTLIB::COS_F128,
3922
3980
RTLIB::COS_PPCF128));
@@ -3926,26 +3984,31 @@ void SelectionDAGLegalize::ConvertNodeToLibcall(SDNode *Node) {
3926
3984
ExpandSinCosLibCall (Node, Results);
3927
3985
break ;
3928
3986
case ISD::FLOG:
3987
+ case ISD::STRICT_FLOG:
3929
3988
Results.push_back (ExpandFPLibCall (Node, RTLIB::LOG_F32, RTLIB::LOG_F64,
3930
3989
RTLIB::LOG_F80, RTLIB::LOG_F128,
3931
3990
RTLIB::LOG_PPCF128));
3932
3991
break ;
3933
3992
case ISD::FLOG2:
3993
+ case ISD::STRICT_FLOG2:
3934
3994
Results.push_back (ExpandFPLibCall (Node, RTLIB::LOG2_F32, RTLIB::LOG2_F64,
3935
3995
RTLIB::LOG2_F80, RTLIB::LOG2_F128,
3936
3996
RTLIB::LOG2_PPCF128));
3937
3997
break ;
3938
3998
case ISD::FLOG10:
3999
+ case ISD::STRICT_FLOG10:
3939
4000
Results.push_back (ExpandFPLibCall (Node, RTLIB::LOG10_F32, RTLIB::LOG10_F64,
3940
4001
RTLIB::LOG10_F80, RTLIB::LOG10_F128,
3941
4002
RTLIB::LOG10_PPCF128));
3942
4003
break ;
3943
4004
case ISD::FEXP:
4005
+ case ISD::STRICT_FEXP:
3944
4006
Results.push_back (ExpandFPLibCall (Node, RTLIB::EXP_F32, RTLIB::EXP_F64,
3945
4007
RTLIB::EXP_F80, RTLIB::EXP_F128,
3946
4008
RTLIB::EXP_PPCF128));
3947
4009
break ;
3948
4010
case ISD::FEXP2:
4011
+ case ISD::STRICT_FEXP2:
3949
4012
Results.push_back (ExpandFPLibCall (Node, RTLIB::EXP2_F32, RTLIB::EXP2_F64,
3950
4013
RTLIB::EXP2_F80, RTLIB::EXP2_F128,
3951
4014
RTLIB::EXP2_PPCF128));
@@ -3966,11 +4029,13 @@ void SelectionDAGLegalize::ConvertNodeToLibcall(SDNode *Node) {
3966
4029
RTLIB::CEIL_PPCF128));
3967
4030
break ;
3968
4031
case ISD::FRINT:
4032
+ case ISD::STRICT_FRINT:
3969
4033
Results.push_back (ExpandFPLibCall (Node, RTLIB::RINT_F32, RTLIB::RINT_F64,
3970
4034
RTLIB::RINT_F80, RTLIB::RINT_F128,
3971
4035
RTLIB::RINT_PPCF128));
3972
4036
break ;
3973
4037
case ISD::FNEARBYINT:
4038
+ case ISD::STRICT_FNEARBYINT:
3974
4039
Results.push_back (ExpandFPLibCall (Node, RTLIB::NEARBYINT_F32,
3975
4040
RTLIB::NEARBYINT_F64,
3976
4041
RTLIB::NEARBYINT_F80,
@@ -3985,11 +4050,13 @@ void SelectionDAGLegalize::ConvertNodeToLibcall(SDNode *Node) {
3985
4050
RTLIB::ROUND_PPCF128));
3986
4051
break ;
3987
4052
case ISD::FPOWI:
4053
+ case ISD::STRICT_FPOWI:
3988
4054
Results.push_back (ExpandFPLibCall (Node, RTLIB::POWI_F32, RTLIB::POWI_F64,
3989
4055
RTLIB::POWI_F80, RTLIB::POWI_F128,
3990
4056
RTLIB::POWI_PPCF128));
3991
4057
break ;
3992
4058
case ISD::FPOW:
4059
+ case ISD::STRICT_FPOW:
3993
4060
Results.push_back (ExpandFPLibCall (Node, RTLIB::POW_F32, RTLIB::POW_F64,
3994
4061
RTLIB::POW_F80, RTLIB::POW_F128,
3995
4062
RTLIB::POW_PPCF128));
0 commit comments