From b905a6b1c4aca04d3ecd7ae1776af58364b10bcb Mon Sep 17 00:00:00 2001 From: James Y Knight Date: Tue, 29 Dec 2015 04:00:43 +0000 Subject: [PATCH] [TrailingObjects] Dynamically realign under-aligned trailing objects. Previously, the code enforced non-decreasing alignment of each trailing type. However, it's easy enough to allow for realignment as needed, and thus avoid the developer having to think about the possiblilities for alignment requirements on all architectures. (E.g. on Linux/x86, a struct with an int64 member is 4-byte aligned, while on other 32-bit archs -- and even with other OSes on x86 -- it has 8-byte alignment. This sort of thing is irritating to have to manually deal with.) git-svn-id: https://llvm.org/svn/llvm-project/llvm/trunk@256533 91177308-0d34-0410-b5e6-96231b3b80d8 --- include/llvm/Support/TrailingObjects.h | 88 ++++++++++++----------- unittests/Support/TrailingObjectsTest.cpp | 17 +++++ 2 files changed, 65 insertions(+), 40 deletions(-) diff --git a/include/llvm/Support/TrailingObjects.h b/include/llvm/Support/TrailingObjects.h index 1561e62261d..8529746eecc 100644 --- a/include/llvm/Support/TrailingObjects.h +++ b/include/llvm/Support/TrailingObjects.h @@ -16,15 +16,14 @@ /// /// The TrailingObject template abstracts away the reinterpret_cast, /// pointer arithmetic, and size calculations used for the allocation -/// and access of appended arrays of objects, as well as asserts that -/// the alignment of the classes involved are appropriate for the -/// usage. Additionally, it ensures that the base type is final -- -/// deriving from a class that expects data appended immediately after -/// it is typically not safe. +/// and access of appended arrays of objects, and takes care that they +/// are all allocated at their required alignment. Additionally, it +/// ensures that the base type is final -- deriving from a class that +/// expects data appended immediately after it is typically not safe. /// /// Users are expected to derive from this template, and provide -/// numTrailingObjects implementations for each trailing type, -/// e.g. like this sample: +/// numTrailingObjects implementations for each trailing type except +/// the last, e.g. like this sample: /// /// \code /// class VarLengthObj : private TrailingObjects { @@ -32,9 +31,6 @@ /// /// unsigned NumInts, NumDoubles; /// size_t numTrailingObjects(OverloadToken) const { return NumInts; } -/// size_t numTrailingObjects(OverloadToken) const { -/// return NumDoubles; -/// } /// }; /// \endcode /// @@ -51,11 +47,12 @@ #ifndef LLVM_SUPPORT_TRAILINGOBJECTS_H #define LLVM_SUPPORT_TRAILINGOBJECTS_H -#include -#include #include "llvm/Support/AlignOf.h" #include "llvm/Support/Compiler.h" +#include "llvm/Support/MathExtras.h" #include "llvm/Support/type_traits.h" +#include +#include namespace llvm { @@ -149,13 +146,8 @@ struct TrailingObjectsImpl::Alignment >= - llvm::AlignOf::Alignment, - "A trailing object requires more alignment than the previous " - "trailing object provides"); - - ParentType::verifyTrailingObjectsAssertions(); + static LLVM_CONSTEXPR bool requiresRealignment() { + return llvm::AlignOf::Alignment < llvm::AlignOf::Alignment; } // These two functions are helper functions for @@ -170,30 +162,45 @@ struct TrailingObjectsImpl) { - return reinterpret_cast( - TopTrailingObj::getTrailingObjectsImpl( - Obj, TrailingObjectsBase::OverloadToken()) + - TopTrailingObj::callNumTrailingObjects( - Obj, TrailingObjectsBase::OverloadToken())); + auto *Ptr = TopTrailingObj::getTrailingObjectsImpl( + Obj, TrailingObjectsBase::OverloadToken()) + + TopTrailingObj::callNumTrailingObjects( + Obj, TrailingObjectsBase::OverloadToken()); + + if (requiresRealignment()) + return reinterpret_cast( + llvm::alignAddr(Ptr, llvm::alignOf())); + else + return reinterpret_cast(Ptr); } static NextTy * getTrailingObjectsImpl(BaseTy *Obj, TrailingObjectsBase::OverloadToken) { - return reinterpret_cast( - TopTrailingObj::getTrailingObjectsImpl( - Obj, TrailingObjectsBase::OverloadToken()) + - TopTrailingObj::callNumTrailingObjects( - Obj, TrailingObjectsBase::OverloadToken())); + auto *Ptr = TopTrailingObj::getTrailingObjectsImpl( + Obj, TrailingObjectsBase::OverloadToken()) + + TopTrailingObj::callNumTrailingObjects( + Obj, TrailingObjectsBase::OverloadToken()); + + if (requiresRealignment()) + return reinterpret_cast( + llvm::alignAddr(Ptr, llvm::alignOf())); + else + return reinterpret_cast(Ptr); } // Helper function for TrailingObjects::additionalSizeToAlloc: this // function recurses to superclasses, each of which requires one // fewer size_t argument, and adds its own size. static LLVM_CONSTEXPR size_t additionalSizeToAllocImpl( - size_t Count1, + size_t SizeSoFar, size_t Count1, typename ExtractSecondType::type... MoreCounts) { - return sizeof(NextTy) * Count1 + additionalSizeToAllocImpl(MoreCounts...); + return additionalSizeToAllocImpl( + (requiresRealignment() + ? llvm::RoundUpToAlignment(SizeSoFar, llvm::alignOf()) + : SizeSoFar) + + sizeof(NextTy) * Count1, + MoreCounts...); } }; @@ -207,9 +214,11 @@ struct TrailingObjectsImpl // up the inheritance chain to subclasses. static void getTrailingObjectsImpl(); - static LLVM_CONSTEXPR size_t additionalSizeToAllocImpl() { return 0; } + static LLVM_CONSTEXPR size_t additionalSizeToAllocImpl(size_t SizeSoFar) { + return SizeSoFar; + } - static void verifyTrailingObjectsAssertions() {} + template static void verifyTrailingObjectsAlignment() {} }; } // end namespace trailing_objects_internal @@ -238,15 +247,14 @@ class TrailingObjects : private trailing_objects_internal::TrailingObjectsImpl< using ParentType::getTrailingObjectsImpl; - // Contains static_assert statements for the alignment of the - // types. Must not be at class-level, because BaseTy isn't complete - // at class instantiation time, but will be by the time this - // function is instantiated. Recurses through the superclasses. + // This function contains only a static_assert BaseTy is final. The + // static_assert must be in a function, and not at class-level + // because BaseTy isn't complete at class instantiation time, but + // will be by the time this function is instantiated. static void verifyTrailingObjectsAssertions() { #ifdef LLVM_IS_FINAL static_assert(LLVM_IS_FINAL(BaseTy), "BaseTy must be final."); #endif - ParentType::verifyTrailingObjectsAssertions(); } // These two methods are the base of the recursion for this method. @@ -320,7 +328,7 @@ public: additionalSizeToAlloc( typename trailing_objects_internal::ExtractSecondType< TrailingTys, size_t>::type... Counts) { - return ParentType::additionalSizeToAllocImpl(Counts...); + return ParentType::additionalSizeToAllocImpl(0, Counts...); } /// Returns the total size of an object if it were allocated with the @@ -332,7 +340,7 @@ public: std::is_same, Foo>::value, size_t>::type totalSizeToAlloc(typename trailing_objects_internal::ExtractSecondType< TrailingTys, size_t>::type... Counts) { - return sizeof(BaseTy) + ParentType::additionalSizeToAllocImpl(Counts...); + return sizeof(BaseTy) + ParentType::additionalSizeToAllocImpl(0, Counts...); } }; diff --git a/unittests/Support/TrailingObjectsTest.cpp b/unittests/Support/TrailingObjectsTest.cpp index 4c05d665f98..866ff1e6e88 100644 --- a/unittests/Support/TrailingObjectsTest.cpp +++ b/unittests/Support/TrailingObjectsTest.cpp @@ -175,4 +175,21 @@ TEST(TrailingObjects, ThreeArg) { reinterpret_cast(reinterpret_cast(C + 1) + 1) + 1)); } + +class Class4 final : public TrailingObjects { + friend TrailingObjects; + size_t numTrailingObjects(OverloadToken) const { return 1; } +}; + +TEST(TrailingObjects, Realignment) { + EXPECT_EQ((Class4::additionalSizeToAlloc(1, 1)), + llvm::RoundUpToAlignment(sizeof(long) + 1, llvm::alignOf())); + EXPECT_EQ(sizeof(Class4), llvm::RoundUpToAlignment(1, llvm::alignOf())); + std::unique_ptr P(new char[1000]); + Class4 *C = reinterpret_cast(P.get()); + EXPECT_EQ(C->getTrailingObjects(), reinterpret_cast(C + 1)); + EXPECT_EQ(C->getTrailingObjects(), + reinterpret_cast(llvm::alignAddr( + reinterpret_cast(C + 1) + 1, llvm::alignOf()))); +} } -- 2.34.1