RFR: 8346916: [REDO] align_up has potential overflow [v5]
Casper Norrbin
cnorrbin at openjdk.org
Tue Mar 4 16:08:53 UTC 2025
On Tue, 4 Mar 2025 09:29:48 GMT, Kim Barrett <kbarrett at openjdk.org> wrote:
>> src/hotspot/share/utilities/align.hpp line 96:
>>
>>> 94:
>>> 95: template <typename T, typename A>
>>> 96: inline T* align_up_or_null(T* ptr, A alignment) {
>>
>> Rather than specialized align_up variants, I think better and more generally
>> usable would be a predicate for testing whether align_up is safe. Something
>> like a `bool can_align_up(value, alignment)` function.
>>
>> I'm happy to see align_up_or_min removed from this PR for that reason, as well
>> as because it has usability issues. Is a min-valued return indicating failure
>> to align? Or was it just the argument is min-valued and already aligned?
>> Consider passing an unsigned zero value as the value to be aligned.
>>
>> This seems to work:
>>
>> diff --git a/src/hotspot/share/utilities/align.hpp b/src/hotspot/share/utilities/align.hpp
>> index b67e61036a0..d73e8e086ca 100644
>> --- a/src/hotspot/share/utilities/align.hpp
>> +++ b/src/hotspot/share/utilities/align.hpp
>> @@ -30,6 +30,8 @@
>> #include "utilities/debug.hpp"
>> #include "utilities/globalDefinitions.hpp"
>> #include "utilities/powerOfTwo.hpp"
>> +
>> +#include <limits>
>> #include <type_traits>
>>
>> // Compute mask to use for aligning to or testing alignment.
>> @@ -70,6 +72,17 @@ constexpr T align_down(T size, A alignment) {
>> return result;
>> }
>>
>> +template<typename T, typename A, ENABLE_IF(std::is_integral<T>::value)>
>> +constexpr bool can_align_up(T size, A alignment) {
>> + return align_down(std::numeric_limits<T>::max(), alignment) >= size;
>> +}
>> +
>> +template<typename A>
>> +inline bool can_align_up(const void* p, A alignment) {
>> + static_assert(sizeof(p) == sizeof(uintptr_t), "assumption");
>> + return can_align_up(reinterpret_cast<uintptr_t>(p), alignment);
>> +}
>> +
>> template<typename T, typename A, ENABLE_IF(std::is_integral<T>::value)>
>> constexpr T align_up(T size, A alignment) {
>> T adjusted = checked_cast<T>(size + alignment_mask(alignment));
>> diff --git a/test/hotspot/gtest/utilities/test_align.cpp b/test/hotspot/gtest/utilities/test_align.cpp
>> index 3c03fd5f24d..2112d43feab 100644
>> --- a/test/hotspot/gtest/utilities/test_align.cpp
>> +++ b/test/hotspot/gtest/utilities/test_align.cpp
>> @@ -27,6 +27,51 @@
>> #include "unittest.hpp"
>>
>> #include <limits>
>> +#include <type_traits>
>> +
>> +template<typename T, typename A>
>> +static constexpr void test_can_align_up() {
>> + int alignment_value = 4;
>> + int small_value = 63;
>> + A alignment = static_cast<A>(alignment_value);
>> +
>> + EXPECT_TRUE(can_align_up...
>
> I forgot to add any description comments for can_align_up. And align_up description should say can_align_up
> is a precondition.
I removed `align_up_or_null` in favour of `can_align_up`, and I agree that it feels more usable.
Instead of `align_up_or_null`, we can now either return early or set a different value if `can_align_up` fails. I updated the files here and to me it feels like an improvement.
-------------
PR Review Comment: https://git.openjdk.org/jdk/pull/23711#discussion_r1979762563
More information about the hotspot-dev
mailing list