Quuxplusone created this revision. Quuxplusone added reviewers: rsmith, espindola, DiggerLin, ahatanak, reames. Herald added subscribers: cfe-commits, rupprecht, dexonsmith, MaskRay, hiraditya. Herald added a reviewer: jhenderson. Herald added a project: clang.
I wrote a Clang warning [not pictured] to diagnose any use of `T(x)` which was not equivalent to `static_cast<T>(x)`. Then I built Clang with that warning turned on. Here are all of the instances of "unsafe" `T(x)` that it identified in the codebase. Every single one is a cast from some pointer type to `uintptr_t`. N.B.: The code in `llvm/IR/SymbolTableListTraits.h` looks absolutely crazy, and this will be the first time it's been touched in over a decade. If this refactor is acceptable, I'll need someone to land it for me, as I don't have permissions. Repository: rG LLVM Github Monorepo https://reviews.llvm.org/D76572 Files: clang/lib/CodeGen/CGCall.h llvm/include/llvm/IR/SymbolTableListTraits.h llvm/include/llvm/Object/Binary.h llvm/lib/CodeGen/AsmPrinter/OcamlGCPrinter.cpp llvm/lib/Object/COFFObjectFile.cpp llvm/lib/Object/ELFObjectFile.cpp llvm/lib/Object/XCOFFObjectFile.cpp llvm/lib/Target/Hexagon/HexagonCommonGEP.cpp llvm/lib/Transforms/Instrumentation/DataFlowSanitizer.cpp llvm/tools/llvm-readobj/ELFDumper.cpp
Index: llvm/tools/llvm-readobj/ELFDumper.cpp =================================================================== --- llvm/tools/llvm-readobj/ELFDumper.cpp +++ llvm/tools/llvm-readobj/ELFDumper.cpp @@ -443,7 +443,7 @@ const ELFFile<ELFT> *Obj = ObjF->getELFFile(); unsigned SecNdx = Sec - &cantFail(Obj->sections()).front(); - if (uintptr_t(Obj->base() + Sec->sh_offset) % sizeof(uint16_t) != 0) + if (reinterpret_cast<uintptr_t>(Obj->base() + Sec->sh_offset) % sizeof(uint16_t) != 0) return createError("the SHT_GNU_versym section with index " + Twine(SecNdx) + " is misaligned"); @@ -522,7 +522,7 @@ Twine(SecNdx) + ": version definition " + Twine(I) + " goes past the end of the section"); - if (uintptr_t(VerdefBuf) % sizeof(uint32_t) != 0) + if (reinterpret_cast<uintptr_t>(VerdefBuf) % sizeof(uint32_t) != 0) return createError( "invalid SHT_GNU_verdef section with index " + Twine(SecNdx) + ": found a misaligned version definition entry at offset 0x" + @@ -545,7 +545,7 @@ const uint8_t *VerdauxBuf = VerdefBuf + D->vd_aux; for (unsigned J = 0; J < D->vd_cnt; ++J) { - if (uintptr_t(VerdauxBuf) % sizeof(uint32_t) != 0) + if (reinterpret_cast<uintptr_t>(VerdauxBuf) % sizeof(uint32_t) != 0) return createError("invalid SHT_GNU_verdef section with index " + Twine(SecNdx) + ": found a misaligned auxiliary entry at offset 0x" + @@ -597,7 +597,7 @@ Twine(SecNdx) + ": version dependency " + Twine(I) + " goes past the end of the section"); - if (uintptr_t(VerneedBuf) % sizeof(uint32_t) != 0) + if (reinterpret_cast<uintptr_t>(VerneedBuf) % sizeof(uint32_t) != 0) return createError( "invalid SHT_GNU_verneed section with index " + Twine(SecNdx) + ": found a misaligned version dependency entry at offset 0x" + @@ -624,7 +624,7 @@ const uint8_t *VernauxBuf = VerneedBuf + Verneed->vn_aux; for (unsigned J = 0; J < Verneed->vn_cnt; ++J) { - if (uintptr_t(VernauxBuf) % sizeof(uint32_t) != 0) + if (reinterpret_cast<uintptr_t>(VernauxBuf) % sizeof(uint32_t) != 0) return createError("invalid SHT_GNU_verneed section with index " + Twine(SecNdx) + ": found a misaligned auxiliary entry at offset 0x" + Index: llvm/lib/Transforms/Instrumentation/DataFlowSanitizer.cpp =================================================================== --- llvm/lib/Transforms/Instrumentation/DataFlowSanitizer.cpp +++ llvm/lib/Transforms/Instrumentation/DataFlowSanitizer.cpp @@ -619,14 +619,14 @@ ArgTLS = nullptr; GetArgTLSTy = FunctionType::get(PointerType::getUnqual(ArgTLSTy), false); GetArgTLS = ConstantExpr::getIntToPtr( - ConstantInt::get(IntptrTy, uintptr_t(GetArgTLSPtr)), + ConstantInt::get(IntptrTy, reinterpret_cast<uintptr_t>(GetArgTLSPtr)), PointerType::getUnqual(GetArgTLSTy)); } if (GetRetvalTLSPtr) { RetvalTLS = nullptr; GetRetvalTLSTy = FunctionType::get(PointerType::getUnqual(ShadowTy), false); GetRetvalTLS = ConstantExpr::getIntToPtr( - ConstantInt::get(IntptrTy, uintptr_t(GetRetvalTLSPtr)), + ConstantInt::get(IntptrTy, reinterpret_cast<uintptr_t>(GetRetvalTLSPtr)), PointerType::getUnqual(GetRetvalTLSTy)); } Index: llvm/lib/Target/Hexagon/HexagonCommonGEP.cpp =================================================================== --- llvm/lib/Target/Hexagon/HexagonCommonGEP.cpp +++ llvm/lib/Target/Hexagon/HexagonCommonGEP.cpp @@ -482,7 +482,8 @@ // determining equality. The only purpose of the ordering is to eliminate // duplication due to the commutativity of equality/non-equality. static NodePair node_pair(GepNode *N1, GepNode *N2) { - uintptr_t P1 = uintptr_t(N1), P2 = uintptr_t(N2); + uintptr_t P1 = reinterpret_cast<uintptr_t>(N1); + uintptr_t P2 = reinterpret_cast<uintptr_t>(N2); if (P1 <= P2) return std::make_pair(N1, N2); return std::make_pair(N2, N1); Index: llvm/lib/Object/XCOFFObjectFile.cpp =================================================================== --- llvm/lib/Object/XCOFFObjectFile.cpp +++ llvm/lib/Object/XCOFFObjectFile.cpp @@ -24,7 +24,7 @@ template <typename T> static Expected<const T *> getObject(MemoryBufferRef M, const void *Ptr, const uint64_t Size = sizeof(T)) { - uintptr_t Addr = uintptr_t(Ptr); + uintptr_t Addr = reinterpret_cast<uintptr_t>(Ptr); if (std::error_code EC = Binary::checkOffset(M, Addr, Size)) return errorCodeToError(EC); return reinterpret_cast<const T *>(Addr); @@ -276,7 +276,7 @@ const uint8_t * ContentStart = base() + OffsetToRaw; uint64_t SectionSize = getSectionSize(Sec); - if (checkOffset(Data, uintptr_t(ContentStart), SectionSize)) + if (checkOffset(Data, reinterpret_cast<uintptr_t>(ContentStart), SectionSize)) return make_error<BinaryError>(); return makeArrayRef(ContentStart,SectionSize); Index: llvm/lib/Object/ELFObjectFile.cpp =================================================================== --- llvm/lib/Object/ELFObjectFile.cpp +++ llvm/lib/Object/ELFObjectFile.cpp @@ -71,7 +71,7 @@ std::pair<unsigned char, unsigned char> Ident = getElfArchType(Obj.getBuffer()); std::size_t MaxAlignment = - 1ULL << countTrailingZeros(uintptr_t(Obj.getBufferStart())); + 1ULL << countTrailingZeros(reinterpret_cast<uintptr_t>(Obj.getBufferStart())); if (MaxAlignment < 2) return createError("Insufficient alignment"); Index: llvm/lib/Object/COFFObjectFile.cpp =================================================================== --- llvm/lib/Object/COFFObjectFile.cpp +++ llvm/lib/Object/COFFObjectFile.cpp @@ -57,7 +57,7 @@ static std::error_code getObject(const T *&Obj, MemoryBufferRef M, const void *Ptr, const uint64_t Size = sizeof(T)) { - uintptr_t Addr = uintptr_t(Ptr); + uintptr_t Addr = reinterpret_cast<uintptr_t>(Ptr); if (std::error_code EC = Binary::checkOffset(M, Addr, Size)) return EC; Obj = reinterpret_cast<const T *>(Addr); @@ -330,7 +330,7 @@ unsigned COFFObjectFile::getSectionID(SectionRef Sec) const { uintptr_t Offset = - uintptr_t(Sec.getRawDataRefImpl().p) - uintptr_t(SectionTable); + reinterpret_cast<uintptr_t>(Sec.getRawDataRefImpl().p) - reinterpret_cast<uintptr_t>(SectionTable); assert((Offset % sizeof(coff_section)) == 0); return (Offset / sizeof(coff_section)) + 1; } @@ -371,7 +371,7 @@ // relocations. begin++; } - if (Binary::checkOffset(M, uintptr_t(begin), + if (Binary::checkOffset(M, reinterpret_cast<uintptr_t>(begin), sizeof(coff_relocation) * NumRelocs)) return nullptr; return begin; @@ -461,7 +461,7 @@ uint32_t SectionEnd = Section->VirtualAddress + Section->VirtualSize; if (SectionStart <= Addr && Addr < SectionEnd) { uint32_t Offset = Addr - SectionStart; - Res = uintptr_t(base()) + Section->PointerToRawData + Offset; + Res = reinterpret_cast<uintptr_t>(base()) + Section->PointerToRawData + Offset; return std::error_code(); } } @@ -480,7 +480,7 @@ if (SectionStart <= RVA && OffsetIntoSection < Section->VirtualSize && Size <= Section->VirtualSize - OffsetIntoSection) { uintptr_t Begin = - uintptr_t(base()) + Section->PointerToRawData + OffsetIntoSection; + reinterpret_cast<uintptr_t>(base()) + Section->PointerToRawData + OffsetIntoSection; Contents = ArrayRef<uint8_t>(reinterpret_cast<const uint8_t *>(Begin), Size); return std::error_code(); @@ -1105,7 +1105,7 @@ // The only thing that we need to verify is that the contents is contained // within the file bounds. We don't need to make sure it doesn't cover other // data, as there's nothing that says that is not allowed. - uintptr_t ConStart = uintptr_t(base()) + Sec->PointerToRawData; + uintptr_t ConStart = reinterpret_cast<uintptr_t>(base()) + Sec->PointerToRawData; uint32_t SectionSize = getSectionSize(Sec); if (checkOffset(Data, ConStart, SectionSize)) return make_error<BinaryError>(); Index: llvm/lib/CodeGen/AsmPrinter/OcamlGCPrinter.cpp =================================================================== --- llvm/lib/CodeGen/AsmPrinter/OcamlGCPrinter.cpp +++ llvm/lib/CodeGen/AsmPrinter/OcamlGCPrinter.cpp @@ -147,7 +147,7 @@ "Frame size " + Twine(FrameSize) + ">= 65536.\n" "(" + - Twine(uintptr_t(&FI)) + ")"); + Twine(reinterpret_cast<uintptr_t>(&FI)) + ")"); } AP.OutStreamer->AddComment("live roots for " + Index: llvm/include/llvm/Object/Binary.h =================================================================== --- llvm/include/llvm/Object/Binary.h +++ llvm/include/llvm/Object/Binary.h @@ -163,8 +163,8 @@ static std::error_code checkOffset(MemoryBufferRef M, uintptr_t Addr, const uint64_t Size) { if (Addr + Size < Addr || Addr + Size < Size || - Addr + Size > uintptr_t(M.getBufferEnd()) || - Addr < uintptr_t(M.getBufferStart())) { + Addr + Size > reinterpret_cast<uintptr_t>(M.getBufferEnd()) || + Addr < reinterpret_cast<uintptr_t>(M.getBufferStart())) { return object_error::unexpected_eof; } return std::error_code(); Index: llvm/include/llvm/IR/SymbolTableListTraits.h =================================================================== --- llvm/include/llvm/IR/SymbolTableListTraits.h +++ llvm/include/llvm/IR/SymbolTableListTraits.h @@ -76,9 +76,9 @@ /// getListOwner - Return the object that owns this list. If this is a list /// of instructions, it returns the BasicBlock that owns them. ItemParentClass *getListOwner() { - size_t Offset(size_t(&((ItemParentClass*)nullptr->*ItemParentClass:: - getSublistAccess(static_cast<ValueSubClass*>(nullptr))))); - ListTy *Anchor(static_cast<ListTy *>(this)); + size_t Offset = reinterpret_cast<size_t>(&((ItemParentClass*)nullptr->*ItemParentClass:: + getSublistAccess(static_cast<ValueSubClass*>(nullptr)))); + ListTy *Anchor = static_cast<ListTy *>(this); return reinterpret_cast<ItemParentClass*>(reinterpret_cast<char*>(Anchor)- Offset); } Index: clang/lib/CodeGen/CGCall.h =================================================================== --- clang/lib/CodeGen/CGCall.h +++ clang/lib/CodeGen/CGCall.h @@ -110,7 +110,7 @@ /// Construct a callee. Call this constructor directly when this /// isn't a direct call. CGCallee(const CGCalleeInfo &abstractInfo, llvm::Value *functionPtr) - : KindOrFunctionPointer(SpecialKind(uintptr_t(functionPtr))) { + : KindOrFunctionPointer(SpecialKind(reinterpret_cast<uintptr_t>(functionPtr))) { AbstractInfo = abstractInfo; assert(functionPtr && "configuring callee without function pointer"); assert(functionPtr->getType()->isPointerTy()); @@ -186,7 +186,7 @@ } void setFunctionPointer(llvm::Value *functionPtr) { assert(isOrdinary()); - KindOrFunctionPointer = SpecialKind(uintptr_t(functionPtr)); + KindOrFunctionPointer = SpecialKind(reinterpret_cast<uintptr_t>(functionPtr)); } bool isVirtual() const {
_______________________________________________ cfe-commits mailing list cfe-commits@lists.llvm.org https://lists.llvm.org/cgi-bin/mailman/listinfo/cfe-commits