================ @@ -2732,6 +2732,92 @@ static bool truncateBitfieldValue(EvalInfo &Info, const Expr *E, return true; } +static bool BitcastAPIntToVector(EvalInfo &Info, const VectorType *VTy, + const llvm::APInt &SValInt, + SmallVectorImpl<APValue> &Elts) { + QualType EltTy = VTy->getElementType(); + unsigned NElts = VTy->getNumElements(); + unsigned EltSize = + VTy->isExtVectorBoolType() ? 1 : Info.Ctx.getTypeSize(EltTy); + + if ((NElts * EltSize) % Info.Ctx.getCharWidth() != 0) { + // The vector's size in bits is not a multiple of the target's byte size, + // so its layout is unspecified. For now, we'll simply treat these cases as + // unsupported (this should only be possible with OpenCL bool vectors whose + // element count isn't a multiple of the byte size). + return false; + } + + Elts.reserve(NElts); + bool BigEndian = Info.Ctx.getTargetInfo().isBigEndian(); + if (EltTy->isRealFloatingType()) { + const llvm::fltSemantics &Sem = Info.Ctx.getFloatTypeSemantics(EltTy); + unsigned FloatEltSize = EltSize; + if (&Sem == &APFloat::x87DoubleExtended()) + FloatEltSize = 80; + for (unsigned i = 0; i < NElts; i++) { + llvm::APInt Elt; + if (BigEndian) + Elt = SValInt.rotl(i * EltSize + FloatEltSize).trunc(FloatEltSize); + else + Elt = SValInt.rotr(i * EltSize).trunc(FloatEltSize); + Elts.push_back(APValue(APFloat(Sem, Elt))); + } + } else if (EltTy->isIntegerType()) { + for (unsigned i = 0; i < NElts; i++) { + llvm::APInt Elt; + if (BigEndian) + Elt = SValInt.rotl(i * EltSize + EltSize).zextOrTrunc(EltSize); + else + Elt = SValInt.rotr(i * EltSize).zextOrTrunc(EltSize); + Elts.push_back(APValue(APSInt(Elt, !EltTy->isSignedIntegerType()))); + } + } else { + return false; + } + return true; +} + +static bool BitcastVectorToAPInt(EvalInfo &Info, const VectorType *VTy, + const APValue &SVal, llvm::APInt &Res) { + QualType EltTy = VTy->getElementType(); + unsigned NElts = VTy->getNumElements(); + unsigned EltSize = + VTy->isExtVectorBoolType() ? 1 : Info.Ctx.getTypeSize(EltTy); + unsigned VecSize = Info.Ctx.getTypeSize(VTy); + + if ((NElts * EltSize) % Info.Ctx.getCharWidth() != 0) { + // The vector's size in bits is not a multiple of the target's byte size, + // so its layout is unspecified. For now, we'll simply treat these cases as + // unsupported (this should only be possible with OpenCL bool vectors whose + // element count isn't a multiple of the byte size). + return false; + } + + bool BigEndian = Info.Ctx.getTargetInfo().isBigEndian(); + Res = llvm::APInt::getZero(VecSize); + for (unsigned i = 0; i < SVal.getVectorLength(); i++) { + const APValue &Elt = SVal.getVectorElt(i); + llvm::APInt EltAsInt; ---------------- zygoloid wrote:
Move this out of the loop so that we only allocate storage for it once. (It typically won't fit in 64 bits, so this will perform a heap allocation.) https://github.com/llvm/llvm-project/pull/66894 _______________________________________________ cfe-commits mailing list cfe-commits@lists.llvm.org https://lists.llvm.org/cgi-bin/mailman/listinfo/cfe-commits