2022-11-05 13:08:40 +01:00
|
|
|
//===--- InterpBuiltin.cpp - Interpreter for the constexpr VM ---*- C++ -*-===//
|
|
|
|
//
|
|
|
|
// Part of the LLVM Project, under the Apache License v2.0 with LLVM Exceptions.
|
|
|
|
// See https://llvm.org/LICENSE.txt for license information.
|
|
|
|
// SPDX-License-Identifier: Apache-2.0 WITH LLVM-exception
|
|
|
|
//
|
|
|
|
//===----------------------------------------------------------------------===//
|
|
|
|
#include "Boolean.h"
|
|
|
|
#include "Interp.h"
|
|
|
|
#include "PrimType.h"
|
|
|
|
#include "clang/Basic/Builtins.h"
|
2023-07-15 07:53:48 +02:00
|
|
|
#include "clang/Basic/TargetInfo.h"
|
2022-11-05 13:08:40 +01:00
|
|
|
|
|
|
|
namespace clang {
|
|
|
|
namespace interp {
|
|
|
|
|
2023-07-15 08:24:52 +02:00
|
|
|
template <typename T>
|
|
|
|
static T getParam(const InterpFrame *Frame, unsigned Index) {
|
2023-07-15 15:16:50 +02:00
|
|
|
assert(Frame->getFunction()->getNumParams() > Index);
|
2023-05-04 07:29:57 +02:00
|
|
|
unsigned Offset = Frame->getFunction()->getParamOffset(Index);
|
|
|
|
return Frame->getParam<T>(Offset);
|
|
|
|
}
|
|
|
|
|
2023-07-16 08:58:06 +02:00
|
|
|
/// Peek an integer value from the stack into an APSInt.
|
|
|
|
static APSInt peekToAPSInt(InterpStack &Stk, PrimType T) {
|
|
|
|
APSInt R;
|
|
|
|
INT_TYPE_SWITCH(T, {
|
|
|
|
T Val = Stk.peek<T>();
|
|
|
|
R = APSInt(APInt(T::bitWidth(), static_cast<uint64_t>(Val), T::isSigned()));
|
|
|
|
});
|
|
|
|
|
|
|
|
return R;
|
|
|
|
}
|
|
|
|
|
2023-05-04 07:29:57 +02:00
|
|
|
static bool interp__builtin_strcmp(InterpState &S, CodePtr OpPC,
|
2023-07-14 20:32:16 +02:00
|
|
|
const InterpFrame *Frame) {
|
2023-05-04 07:29:57 +02:00
|
|
|
const Pointer &A = getParam<Pointer>(Frame, 0);
|
|
|
|
const Pointer &B = getParam<Pointer>(Frame, 1);
|
|
|
|
|
|
|
|
if (!CheckLive(S, OpPC, A, AK_Read) || !CheckLive(S, OpPC, B, AK_Read))
|
|
|
|
return false;
|
|
|
|
|
|
|
|
assert(A.getFieldDesc()->isPrimitiveArray());
|
|
|
|
assert(B.getFieldDesc()->isPrimitiveArray());
|
|
|
|
|
|
|
|
unsigned IndexA = A.getIndex();
|
|
|
|
unsigned IndexB = B.getIndex();
|
|
|
|
int32_t Result = 0;
|
|
|
|
for (;; ++IndexA, ++IndexB) {
|
|
|
|
const Pointer &PA = A.atIndex(IndexA);
|
|
|
|
const Pointer &PB = B.atIndex(IndexB);
|
|
|
|
if (!CheckRange(S, OpPC, PA, AK_Read) ||
|
|
|
|
!CheckRange(S, OpPC, PB, AK_Read)) {
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
uint8_t CA = PA.deref<uint8_t>();
|
|
|
|
uint8_t CB = PB.deref<uint8_t>();
|
|
|
|
|
|
|
|
if (CA > CB) {
|
|
|
|
Result = 1;
|
|
|
|
break;
|
|
|
|
} else if (CA < CB) {
|
|
|
|
Result = -1;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
if (CA == 0 || CB == 0)
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
|
|
|
S.Stk.push<Integral<32, true>>(Integral<32, true>::from(Result));
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
2023-07-15 07:53:48 +02:00
|
|
|
static bool interp__builtin_nan(InterpState &S, CodePtr OpPC,
|
|
|
|
const InterpFrame *Frame, const Function *F,
|
|
|
|
bool Signaling) {
|
|
|
|
const Pointer &Arg = getParam<Pointer>(Frame, 0);
|
|
|
|
|
|
|
|
if (!CheckLoad(S, OpPC, Arg))
|
|
|
|
return false;
|
|
|
|
|
|
|
|
assert(Arg.getFieldDesc()->isPrimitiveArray());
|
|
|
|
|
|
|
|
// Convert the given string to an integer using StringRef's API.
|
|
|
|
llvm::APInt Fill;
|
|
|
|
std::string Str;
|
|
|
|
assert(Arg.getNumElems() >= 1);
|
|
|
|
for (unsigned I = 0;; ++I) {
|
|
|
|
const Pointer &Elem = Arg.atIndex(I);
|
|
|
|
|
|
|
|
if (!CheckLoad(S, OpPC, Elem))
|
|
|
|
return false;
|
|
|
|
|
|
|
|
if (Elem.deref<int8_t>() == 0)
|
|
|
|
break;
|
|
|
|
|
|
|
|
Str += Elem.deref<char>();
|
|
|
|
}
|
|
|
|
|
|
|
|
// Treat empty strings as if they were zero.
|
|
|
|
if (Str.empty())
|
|
|
|
Fill = llvm::APInt(32, 0);
|
|
|
|
else if (StringRef(Str).getAsInteger(0, Fill))
|
|
|
|
return false;
|
|
|
|
|
|
|
|
const llvm::fltSemantics &TargetSemantics =
|
|
|
|
S.getCtx().getFloatTypeSemantics(F->getDecl()->getReturnType());
|
|
|
|
|
|
|
|
Floating Result;
|
|
|
|
if (S.getCtx().getTargetInfo().isNan2008()) {
|
|
|
|
if (Signaling)
|
|
|
|
Result = Floating(
|
|
|
|
llvm::APFloat::getSNaN(TargetSemantics, /*Negative=*/false, &Fill));
|
|
|
|
else
|
|
|
|
Result = Floating(
|
|
|
|
llvm::APFloat::getQNaN(TargetSemantics, /*Negative=*/false, &Fill));
|
|
|
|
} else {
|
|
|
|
// Prior to IEEE 754-2008, architectures were allowed to choose whether
|
|
|
|
// the first bit of their significand was set for qNaN or sNaN. MIPS chose
|
|
|
|
// a different encoding to what became a standard in 2008, and for pre-
|
|
|
|
// 2008 revisions, MIPS interpreted sNaN-2008 as qNan and qNaN-2008 as
|
|
|
|
// sNaN. This is now known as "legacy NaN" encoding.
|
|
|
|
if (Signaling)
|
|
|
|
Result = Floating(
|
|
|
|
llvm::APFloat::getQNaN(TargetSemantics, /*Negative=*/false, &Fill));
|
|
|
|
else
|
|
|
|
Result = Floating(
|
|
|
|
llvm::APFloat::getSNaN(TargetSemantics, /*Negative=*/false, &Fill));
|
|
|
|
}
|
|
|
|
|
|
|
|
S.Stk.push<Floating>(Result);
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
2023-07-15 11:54:22 +02:00
|
|
|
static bool interp__builtin_inf(InterpState &S, CodePtr OpPC,
|
|
|
|
const InterpFrame *Frame, const Function *F) {
|
|
|
|
const llvm::fltSemantics &TargetSemantics =
|
|
|
|
S.getCtx().getFloatTypeSemantics(F->getDecl()->getReturnType());
|
|
|
|
|
|
|
|
S.Stk.push<Floating>(Floating::getInf(TargetSemantics));
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
2023-07-18 08:00:13 +02:00
|
|
|
static bool interp__builtin_copysign(InterpState &S, CodePtr OpPC,
|
|
|
|
const InterpFrame *Frame,
|
|
|
|
const Function *F) {
|
|
|
|
const Floating &Arg1 = getParam<Floating>(Frame, 0);
|
|
|
|
const Floating &Arg2 = getParam<Floating>(Frame, 1);
|
|
|
|
|
|
|
|
APFloat Copy = Arg1.getAPFloat();
|
|
|
|
Copy.copySign(Arg2.getAPFloat());
|
|
|
|
S.Stk.push<Floating>(Floating(Copy));
|
|
|
|
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
2023-07-18 08:00:30 +02:00
|
|
|
static bool interp__builtin_fmin(InterpState &S, CodePtr OpPC,
|
|
|
|
const InterpFrame *Frame, const Function *F) {
|
|
|
|
const Floating &LHS = getParam<Floating>(Frame, 0);
|
|
|
|
const Floating &RHS = getParam<Floating>(Frame, 1);
|
|
|
|
|
|
|
|
Floating Result;
|
|
|
|
|
|
|
|
// When comparing zeroes, return -0.0 if one of the zeroes is negative.
|
|
|
|
if (LHS.isZero() && RHS.isZero() && RHS.isNegative())
|
|
|
|
Result = RHS;
|
|
|
|
else if (LHS.isNan() || RHS < LHS)
|
|
|
|
Result = RHS;
|
|
|
|
else
|
|
|
|
Result = LHS;
|
|
|
|
|
|
|
|
S.Stk.push<Floating>(Result);
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
2023-07-28 21:01:05 +02:00
|
|
|
/// Defined as __builtin_isnan(...), to accommodate the fact that it can
|
|
|
|
/// take a float, double, long double, etc.
|
|
|
|
/// But for us, that's all a Floating anyway.
|
|
|
|
static bool interp__builtin_isnan(InterpState &S, CodePtr OpPC,
|
|
|
|
const InterpFrame *Frame, const Function *F) {
|
|
|
|
const Floating &Arg = S.Stk.peek<Floating>();
|
|
|
|
|
|
|
|
S.Stk.push<Integral<32, true>>(Integral<32, true>::from(Arg.isNan()));
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
2023-07-15 17:47:00 +02:00
|
|
|
static bool interp__builtin_isinf(InterpState &S, CodePtr OpPC,
|
|
|
|
const InterpFrame *Frame, const Function *F,
|
|
|
|
bool CheckSign) {
|
|
|
|
const Floating &Arg = S.Stk.peek<Floating>();
|
|
|
|
bool IsInf = Arg.isInf();
|
|
|
|
|
|
|
|
if (CheckSign)
|
|
|
|
S.Stk.push<Integral<32, true>>(
|
|
|
|
Integral<32, true>::from(IsInf ? (Arg.isNegative() ? -1 : 1) : 0));
|
|
|
|
else
|
|
|
|
S.Stk.push<Integral<32, true>>(Integral<32, true>::from(Arg.isInf()));
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
2023-07-15 18:27:56 +02:00
|
|
|
static bool interp__builtin_isfinite(InterpState &S, CodePtr OpPC,
|
|
|
|
const InterpFrame *Frame,
|
|
|
|
const Function *F) {
|
|
|
|
const Floating &Arg = S.Stk.peek<Floating>();
|
|
|
|
|
|
|
|
S.Stk.push<Integral<32, true>>(Integral<32, true>::from(Arg.isFinite()));
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
2023-07-15 18:37:59 +02:00
|
|
|
static bool interp__builtin_isnormal(InterpState &S, CodePtr OpPC,
|
|
|
|
const InterpFrame *Frame,
|
|
|
|
const Function *F) {
|
|
|
|
const Floating &Arg = S.Stk.peek<Floating>();
|
|
|
|
|
|
|
|
S.Stk.push<Integral<32, true>>(Integral<32, true>::from(Arg.isNormal()));
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
2023-07-16 08:58:06 +02:00
|
|
|
/// First parameter to __builtin_isfpclass is the floating value, the
|
|
|
|
/// second one is an integral value.
|
|
|
|
static bool interp__builtin_isfpclass(InterpState &S, CodePtr OpPC,
|
|
|
|
const InterpFrame *Frame,
|
|
|
|
const Function *Func) {
|
|
|
|
const Expr *E = S.Current->getExpr(OpPC);
|
|
|
|
const CallExpr *CE = cast<CallExpr>(E);
|
|
|
|
PrimType FPClassArgT = *S.getContext().classify(CE->getArgs()[1]->getType());
|
|
|
|
APSInt FPClassArg = peekToAPSInt(S.Stk, FPClassArgT);
|
|
|
|
const Floating &F =
|
|
|
|
S.Stk.peek<Floating>(align(primSize(FPClassArgT) + primSize(PT_Float)));
|
|
|
|
|
|
|
|
int32_t Result =
|
|
|
|
static_cast<int32_t>((F.classify() & FPClassArg).getZExtValue());
|
|
|
|
S.Stk.push<Integral<32, true>>(Integral<32, true>::from(Result));
|
|
|
|
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
2023-07-16 10:20:17 +02:00
|
|
|
/// Five int32 values followed by one floating value.
|
|
|
|
static bool interp__builtin_fpclassify(InterpState &S, CodePtr OpPC,
|
|
|
|
const InterpFrame *Frame,
|
|
|
|
const Function *Func) {
|
|
|
|
const Floating &Val = S.Stk.peek<Floating>();
|
|
|
|
|
|
|
|
unsigned Index;
|
|
|
|
switch (Val.getCategory()) {
|
|
|
|
case APFloat::fcNaN:
|
|
|
|
Index = 0;
|
|
|
|
break;
|
|
|
|
case APFloat::fcInfinity:
|
|
|
|
Index = 1;
|
|
|
|
break;
|
|
|
|
case APFloat::fcNormal:
|
|
|
|
Index = Val.isDenormal() ? 3 : 2;
|
|
|
|
break;
|
|
|
|
case APFloat::fcZero:
|
|
|
|
Index = 4;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
|
|
|
// The last argument is first on the stack.
|
|
|
|
unsigned Offset = align(primSize(PT_Float)) +
|
|
|
|
((1 + (4 - Index)) * align(primSize(PT_Sint32)));
|
|
|
|
|
|
|
|
const Integral<32, true> &I = S.Stk.peek<Integral<32, true>>(Offset);
|
|
|
|
S.Stk.push<Integral<32, true>>(I);
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
2023-05-04 07:29:57 +02:00
|
|
|
bool InterpretBuiltin(InterpState &S, CodePtr OpPC, const Function *F) {
|
|
|
|
InterpFrame *Frame = S.Current;
|
2023-01-07 12:05:05 +01:00
|
|
|
APValue Dummy;
|
2022-11-05 13:08:40 +01:00
|
|
|
|
2023-05-04 07:29:57 +02:00
|
|
|
switch (F->getBuiltinID()) {
|
2022-11-05 13:08:40 +01:00
|
|
|
case Builtin::BI__builtin_is_constant_evaluated:
|
|
|
|
S.Stk.push<Boolean>(Boolean::from(S.inConstantContext()));
|
2023-05-11 12:01:17 +02:00
|
|
|
return Ret<PT_Bool>(S, OpPC, Dummy);
|
2023-01-07 12:05:05 +01:00
|
|
|
case Builtin::BI__builtin_assume:
|
2023-05-11 12:01:17 +02:00
|
|
|
return RetVoid(S, OpPC, Dummy);
|
2023-05-04 07:29:57 +02:00
|
|
|
case Builtin::BI__builtin_strcmp:
|
|
|
|
if (interp__builtin_strcmp(S, OpPC, Frame))
|
2023-05-11 12:01:17 +02:00
|
|
|
return Ret<PT_Sint32>(S, OpPC, Dummy);
|
2023-07-15 07:53:48 +02:00
|
|
|
break;
|
|
|
|
case Builtin::BI__builtin_nan:
|
|
|
|
case Builtin::BI__builtin_nanf:
|
|
|
|
case Builtin::BI__builtin_nanl:
|
|
|
|
case Builtin::BI__builtin_nanf16:
|
|
|
|
case Builtin::BI__builtin_nanf128:
|
|
|
|
if (interp__builtin_nan(S, OpPC, Frame, F, /*Signaling=*/false))
|
|
|
|
return Ret<PT_Float>(S, OpPC, Dummy);
|
|
|
|
break;
|
|
|
|
case Builtin::BI__builtin_nans:
|
|
|
|
case Builtin::BI__builtin_nansf:
|
|
|
|
case Builtin::BI__builtin_nansl:
|
|
|
|
case Builtin::BI__builtin_nansf16:
|
|
|
|
case Builtin::BI__builtin_nansf128:
|
|
|
|
if (interp__builtin_nan(S, OpPC, Frame, F, /*Signaling=*/true))
|
|
|
|
return Ret<PT_Float>(S, OpPC, Dummy);
|
|
|
|
break;
|
|
|
|
|
2023-07-15 11:54:22 +02:00
|
|
|
case Builtin::BI__builtin_huge_val:
|
|
|
|
case Builtin::BI__builtin_huge_valf:
|
|
|
|
case Builtin::BI__builtin_huge_vall:
|
|
|
|
case Builtin::BI__builtin_huge_valf16:
|
|
|
|
case Builtin::BI__builtin_huge_valf128:
|
|
|
|
case Builtin::BI__builtin_inf:
|
|
|
|
case Builtin::BI__builtin_inff:
|
|
|
|
case Builtin::BI__builtin_infl:
|
|
|
|
case Builtin::BI__builtin_inff16:
|
|
|
|
case Builtin::BI__builtin_inff128:
|
|
|
|
if (interp__builtin_inf(S, OpPC, Frame, F))
|
|
|
|
return Ret<PT_Float>(S, OpPC, Dummy);
|
|
|
|
break;
|
2023-07-18 08:00:13 +02:00
|
|
|
case Builtin::BI__builtin_copysign:
|
|
|
|
case Builtin::BI__builtin_copysignf:
|
|
|
|
case Builtin::BI__builtin_copysignl:
|
|
|
|
case Builtin::BI__builtin_copysignf128:
|
|
|
|
if (interp__builtin_copysign(S, OpPC, Frame, F))
|
|
|
|
return Ret<PT_Float>(S, OpPC, Dummy);
|
|
|
|
break;
|
2023-07-15 11:54:22 +02:00
|
|
|
|
2023-07-18 08:00:30 +02:00
|
|
|
case Builtin::BI__builtin_fmin:
|
|
|
|
case Builtin::BI__builtin_fminf:
|
|
|
|
case Builtin::BI__builtin_fminl:
|
|
|
|
case Builtin::BI__builtin_fminf16:
|
|
|
|
case Builtin::BI__builtin_fminf128:
|
|
|
|
if (interp__builtin_fmin(S, OpPC, Frame, F))
|
|
|
|
return Ret<PT_Float>(S, OpPC, Dummy);
|
|
|
|
break;
|
|
|
|
|
2023-07-28 21:01:05 +02:00
|
|
|
case Builtin::BI__builtin_isnan:
|
|
|
|
if (interp__builtin_isnan(S, OpPC, Frame, F))
|
|
|
|
return Ret<PT_Sint32>(S, OpPC, Dummy);
|
|
|
|
break;
|
|
|
|
|
2023-07-15 17:47:00 +02:00
|
|
|
case Builtin::BI__builtin_isinf:
|
|
|
|
if (interp__builtin_isinf(S, OpPC, Frame, F, /*Sign=*/false))
|
|
|
|
return Ret<PT_Sint32>(S, OpPC, Dummy);
|
|
|
|
break;
|
|
|
|
|
|
|
|
case Builtin::BI__builtin_isinf_sign:
|
|
|
|
if (interp__builtin_isinf(S, OpPC, Frame, F, /*Sign=*/true))
|
|
|
|
return Ret<PT_Sint32>(S, OpPC, Dummy);
|
|
|
|
break;
|
|
|
|
|
2023-07-15 18:27:56 +02:00
|
|
|
case Builtin::BI__builtin_isfinite:
|
|
|
|
if (interp__builtin_isfinite(S, OpPC, Frame, F))
|
|
|
|
return Ret<PT_Sint32>(S, OpPC, Dummy);
|
|
|
|
break;
|
2023-07-15 18:37:59 +02:00
|
|
|
case Builtin::BI__builtin_isnormal:
|
|
|
|
if (interp__builtin_isnormal(S, OpPC, Frame, F))
|
|
|
|
return Ret<PT_Sint32>(S, OpPC, Dummy);
|
|
|
|
break;
|
2023-07-16 08:58:06 +02:00
|
|
|
case Builtin::BI__builtin_isfpclass:
|
|
|
|
if (interp__builtin_isfpclass(S, OpPC, Frame, F))
|
|
|
|
return Ret<PT_Sint32>(S, OpPC, Dummy);
|
|
|
|
break;
|
2023-07-16 10:20:17 +02:00
|
|
|
case Builtin::BI__builtin_fpclassify:
|
|
|
|
if (interp__builtin_fpclassify(S, OpPC, Frame, F))
|
|
|
|
return Ret<PT_Sint32>(S, OpPC, Dummy);
|
|
|
|
break;
|
2023-07-15 18:27:56 +02:00
|
|
|
|
2023-01-07 12:05:05 +01:00
|
|
|
default:
|
|
|
|
return false;
|
2022-11-05 13:08:40 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
} // namespace interp
|
|
|
|
} // namespace clang
|