Skip to content

Commit

Permalink
]: enable asan checks for tails
Browse files Browse the repository at this point in the history
Summary: In asan mode we should still check that the user didn't provide us invalid addresses.

Reviewed By: yfeldblum

Differential Revision: D64039159
  • Loading branch information
DenisYaroshevskiy authored and facebook-github-bot committed Oct 10, 2024
1 parent 475d3e4 commit 688eef5
Show file tree
Hide file tree
Showing 4 changed files with 93 additions and 7 deletions.
2 changes: 1 addition & 1 deletion folly/algorithm/simd/detail/BUCK
Original file line number Diff line number Diff line change
Expand Up @@ -22,7 +22,7 @@ cpp_library(
"//folly:portability",
"//folly/algorithm/simd:ignore",
"//folly/algorithm/simd:movemask",
"//folly/lang:bits",
"//folly/lang:safe_assert",
],
)

Expand Down
84 changes: 78 additions & 6 deletions folly/algorithm/simd/detail/SimdPlatform.h
Original file line number Diff line number Diff line change
Expand Up @@ -20,7 +20,7 @@
#include <folly/algorithm/simd/Ignore.h>
#include <folly/algorithm/simd/Movemask.h>
#include <folly/algorithm/simd/detail/SimdPlatform.h>
#include <folly/lang/Bits.h>
#include <folly/lang/SafeAssert.h>

#include <array>

Expand Down Expand Up @@ -96,6 +96,9 @@ struct SimdPlatformCommon {
template <typename Ignore>
static bool any(logical_t logical, Ignore ignore);

template <typename Ignore>
static bool all(logical_t logical, Ignore ignore);

/**
* logical operations
**/
Expand All @@ -110,7 +113,7 @@ struct SimdPlatformCommon {
template <typename Platform>
template <typename Ignore>
FOLLY_ERASE auto SimdPlatformCommon<Platform>::loada(
const scalar_t* ptr, Ignore) -> reg_t {
const scalar_t* ptr, [[maybe_unused]] Ignore ignore) -> reg_t {
if constexpr (std::is_same_v<ignore_none, Ignore>) {
// There is not point to aligned load instructions
// on modern cpus. Arm doesn't even have any.
Expand All @@ -122,7 +125,25 @@ FOLLY_ERASE auto SimdPlatformCommon<Platform>::loada(
//
// Here is an explanation from Stephen Canon:
// https://stackoverflow.com/questions/25566302/vectorized-strlen-getting-away-with-reading-unallocated-memory
return unsafeLoadu(ptr, ignore_none{});
if constexpr (!kIsSanitizeAddress) {
return unsafeLoadu(ptr, ignore_none{});
} else {
// If the sanitizers are enabled, we want to trigger the issues.
// We also want to match the garbage values with/without asan,
// so that testing works on the same values as prod.
scalar_t buf[kCardinal];
std::memcpy(
buf + ignore.first,
ptr + ignore.first,
(kCardinal - ignore.first - ignore.last) * sizeof(scalar_t));

auto testAgainst = loadu(buf, ignore_none{});
auto res = unsafeLoadu(ptr, ignore_none{});

// Extra sanity check.
FOLLY_SAFE_DCHECK(all(Platform::equal(res, testAgainst), ignore));
return res;
}
}
}

Expand Down Expand Up @@ -162,6 +183,24 @@ FOLLY_ERASE bool SimdPlatformCommon<Platform>::any(
}
}

template <typename Platform>
template <typename Ignore>
FOLLY_ERASE bool SimdPlatformCommon<Platform>::all(
logical_t logical, Ignore ignore) {
if constexpr (std::is_same_v<Ignore, ignore_none>) {
return Platform::all(logical);
} else {
auto [bits, bitsPerElement] = movemask<scalar_t>(logical, ignore_none{});

auto expected = n_least_significant_bits<decltype(bits)>(
bitsPerElement * (kCardinal - ignore.last));
expected =
clear_n_least_significant_bits(expected, ignore.first * bitsPerElement);

return (bits & expected) == expected;
}
}

template <typename Platform>
FOLLY_ERASE auto SimdPlatformCommon<Platform>::logical_or(
logical_t x, logical_t y) -> logical_t {
Expand All @@ -185,6 +224,8 @@ struct SimdSse42PlatformSpecific {
using reg_t = __m128i;
using logical_t = reg_t;

static constexpr std::size_t kCardinal = sizeof(reg_t) / sizeof(scalar_t);

FOLLY_ERASE
static reg_t loadu(const scalar_t* p) {
return _mm_loadu_si128(reinterpret_cast<const reg_t*>(p));
Expand Down Expand Up @@ -238,7 +279,16 @@ struct SimdSse42PlatformSpecific {
}

FOLLY_ERASE
static bool any(logical_t log) { return movemask<std::uint8_t>(log).first; }
static bool any(logical_t log) { return movemask<scalar_t>(log).first; }

#if 0 // disabled untill we have a test where this is relevant
FOLLY_ERASE
static bool all(logical_t log) {
auto [bits, bitsPerElement] = movemask<scalar_t>(log);
return movemask<scalar_t>(log) ==
n_least_significant_bits<decltype(bits)>(kCardinal * bitsPerElement);
}
#endif
};

#define FOLLY_DETAIL_HAS_SIMD_PLATFORM 1
Expand All @@ -254,6 +304,8 @@ struct SimdAvx2PlatformSpecific {
using reg_t = __m256i;
using logical_t = reg_t;

static constexpr std::size_t kCardinal = sizeof(reg_t) / sizeof(scalar_t);

FOLLY_ERASE
static reg_t loadu(const scalar_t* p) {
return _mm256_loadu_si256(reinterpret_cast<const reg_t*>(p));
Expand Down Expand Up @@ -306,9 +358,16 @@ struct SimdAvx2PlatformSpecific {
}

FOLLY_ERASE
static bool any(logical_t log) {
return simd::movemask<std::uint8_t>(log).first;
static bool any(logical_t log) { return simd::movemask<scalar_t>(log).first; }

#if 0 // disabled untill we have a test where this is relevant
FOLLY_ERASE
static bool all(logical_t log) {
auto [bits, bitsPerElement] = movemask<scalar_t>(log);
return movemask<scalar_t>(log) ==
n_least_significant_bits<decltype(bits)>(kCardinal * bitsPerElement);
}
#endif
};

template <typename T>
Expand Down Expand Up @@ -420,6 +479,19 @@ struct SimdAarch64PlatformSpecific {
auto u64 = bit_cast<uint64x2_t>(u32);
return vgetq_lane_u64(u64, 0);
}

#if 0 // disabled untill we have a test where this is relevant
FOLLY_ERASE
static bool all(logical_t log) {
// Not quite what they did in .Net runtime, but
// should be close.
// https://github.com/dotnet/runtime/pull/75864
auto u32 = bit_cast<uint32x4_t>(log);
u32 = vpminq_u32(u32, u32);
auto u64 = bit_cast<uint64x2_t>(u32);
return u64 == n_least_significant_bits<std::uint64_t>(64);
}
#endif
};

#define FOLLY_DETAIL_HAS_SIMD_PLATFORM 1
Expand Down
1 change: 1 addition & 0 deletions folly/algorithm/simd/test/BUCK
Original file line number Diff line number Diff line change
Expand Up @@ -11,6 +11,7 @@ cpp_unittest(
"//folly/algorithm/simd:contains",
"//folly/algorithm/simd/detail:simd_contains_impl",
"//folly/portability:gtest",
"//folly/test:test_utils",
],
)

Expand Down
13 changes: 13 additions & 0 deletions folly/algorithm/simd/test/ContainsTest.cpp
Original file line number Diff line number Diff line change
Expand Up @@ -19,12 +19,15 @@
#include <folly/algorithm/simd/detail/ContainsImpl.h>

#include <folly/portability/GTest.h>
#include <folly/test/TestUtils.h>

#include <list>
#include <vector>

namespace folly::simd {

//

static_assert( //
!std::is_invocable_v< //
contains_fn,
Expand Down Expand Up @@ -158,4 +161,14 @@ TEST_F(ContainsTestSpeicalCases, Pointers) {
EXPECT_FALSE(folly::simd::contains(ptrs, &ints[2]));
}

TEST_F(ContainsTestSpeicalCases, AsanShouldFindCrash) {
SKIP_IF(folly::kIsSanitizeAddress);

std::vector<int> v;
v.resize(3);
std::span<int> s(v.begin() + 1, v.begin() + 4);
EXPECT_DEATH(
(folly::simd::contains(s, 0)), "AddressSanitizer: heap-buffer-overflow");
}

} // namespace folly::simd

0 comments on commit 688eef5

Please sign in to comment.