llvmbot wrote:
<!--LLVM PR SUMMARY COMMENT--> @llvm/pr-subscribers-clang Author: Timm Baeder (tbaederr) <details> <summary>Changes</summary> This is a little easier to work with since we are guaranteed that the item type of the vector is byte sized and not something else. --- Full diff: https://github.com/llvm/llvm-project/pull/114677.diff 1 Files Affected: - (modified) clang/lib/AST/ByteCode/InterpBuiltinBitCast.cpp (+20-23) ``````````diff diff --git a/clang/lib/AST/ByteCode/InterpBuiltinBitCast.cpp b/clang/lib/AST/ByteCode/InterpBuiltinBitCast.cpp index fde2c6d9b11ac8..1acd49de307b96 100644 --- a/clang/lib/AST/ByteCode/InterpBuiltinBitCast.cpp +++ b/clang/lib/AST/ByteCode/InterpBuiltinBitCast.cpp @@ -83,39 +83,33 @@ static void swapBytes(std::byte *M, size_t N) { /// have indeterminate value. /// All offsets are in bits. struct BitcastBuffer { - llvm::BitVector Data; + size_t SizeInBits = 0; + llvm::SmallVector<std::byte> Data; BitcastBuffer() = default; - size_t size() const { return Data.size(); } + size_t size() const { return SizeInBits; } - const std::byte *data() const { - unsigned NBytes = Data.size() / 8; - unsigned BitVectorWordSize = sizeof(uintptr_t); - bool FullWord = (NBytes % BitVectorWordSize == 0); - - // llvm::BitVector uses 64-bit fields internally, so when we have - // fewer bytes than that, we need to compensate for that on - // big endian hosts. - unsigned DataPlus; - if (llvm::sys::IsBigEndianHost) - DataPlus = BitVectorWordSize - (NBytes % BitVectorWordSize); - else - DataPlus = 0; - - return reinterpret_cast<const std::byte *>(Data.getData().data()) + - (FullWord ? 0 : DataPlus); - } + const std::byte *data() const { return Data.data(); } bool allInitialized() const { // FIXME: Implement. return true; } + bool atByteBoundary() const { return (Data.size() * 8) == SizeInBits; } + + void pushBit(bool Value) { + if (atByteBoundary()) + Data.push_back(std::byte{0}); + + if (Value) + Data.back() |= (std::byte{1} << (SizeInBits % 8)); + ++SizeInBits; + } + void pushData(const std::byte *data, size_t BitOffset, size_t BitWidth, bool BigEndianTarget) { - Data.reserve(BitOffset + BitWidth); - bool OnlyFullBytes = BitWidth % 8 == 0; unsigned NBytes = BitWidth / 8; @@ -125,7 +119,7 @@ struct BitcastBuffer { std::byte B = BigEndianTarget ? data[NBytes - OnlyFullBytes - I] : data[I]; for (unsigned X = 0; X != 8; ++X) { - Data.push_back(bitof(B, X)); + pushBit(bitof(B, X)); ++BitsHandled; } } @@ -137,7 +131,7 @@ struct BitcastBuffer { assert((BitWidth - BitsHandled) < 8); std::byte B = BigEndianTarget ? data[0] : data[NBytes]; for (size_t I = 0, E = (BitWidth - BitsHandled); I != E; ++I) { - Data.push_back(bitof(B, I)); + pushBit(bitof(B, I)); ++BitsHandled; } @@ -363,5 +357,8 @@ bool clang::interp::DoBitCast(InterpState &S, CodePtr OpPC, const Pointer &Ptr, HasIndeterminateBits = !Buffer.allInitialized(); std::memcpy(Buff, Buffer.data(), BuffSize); + if (llvm::sys::IsBigEndianHost) + swapBytes(Buff, BuffSize); + return Success; } `````````` </details> https://github.com/llvm/llvm-project/pull/114677 _______________________________________________ cfe-commits mailing list cfe-commits@lists.llvm.org https://lists.llvm.org/cgi-bin/mailman/listinfo/cfe-commits