Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

GH-41813: [C++] Fix avx2 gather offset larger than 2GB in CompareColumnsToRows #42188

Merged
merged 24 commits into from
Jun 25, 2024

Conversation

zanmato1984
Copy link
Contributor

@zanmato1984 zanmato1984 commented Jun 17, 2024

Rationale for this change

AVX2 intrinsics _mm256_i32gather_epi32/_mm256_i32gather_epi64 are used in CompareColumnsToRows API, and treat the vindex as signed integer. In our row table implementation, we use uint32_t to represent the offset within the row table. When a offset is larger than (0x80000000, or 2GB), the aforementioned intrinsics will treat it as negative offset and gather the data from undesired address. More details please see #41813 (comment).

Considering there is no unsigned-32bit-offset or 64bit-offset counterparts of those intrinsics in AVX2, this issue can be simply mitigated by translating the base address and the offset:

new_base = base + 0x80000000;
new_offset = offset - 0x80000000;

What changes are included in this PR?

Fix and UT that reproduces the issue.

Are these changes tested?

UT included.

Are there any user-facing changes?

None.

Copy link

⚠️ GitHub issue #41813 has been automatically assigned in GitHub to PR creator.

@zanmato1984
Copy link
Contributor Author

cc @pitrou @amoeba @mrd0ll4r

@zanmato1984 zanmato1984 added the Critical Fix Bugfixes for security vulnerabilities, crashes, or invalid data. label Jun 17, 2024
@github-actions github-actions bot added awaiting committer review Awaiting committer review and removed awaiting review Awaiting review labels Jun 17, 2024
@amoeba
Copy link
Member

amoeba commented Jun 17, 2024

Hi @zanmato1984, thanks for your work on this. I'm hoping others can review the implementation but I did just check that the new test passes (it does) and also fixes the original issue (it does). 👍

@zanmato1984
Copy link
Contributor Author

Hi @zanmato1984, thanks for your work on this. I'm hoping others can review the implementation but I did just check that the new test passes (it does) and also fixes the original issue (it does). 👍

Thank you @amoeba for verifying, and the help on reproducing the issue!

@mrd0ll4r
Copy link

I can't give much feedback or test this out, unfortunately. But I'm very thankful for you all looking into this!

@FreekPaans
Copy link

FreekPaans commented Jun 20, 2024

Ran into this issue when I was debugging my own issue where running a group_by/aggregate on a table with null columns was failing to group some keys, i.e. some group key value tuples were duplicated in the result.

Why I mention it here:

  • My investigation also pointed to a problem with CompareColumnsToRows, but I don't fully understand the code.
  • Disabling AVX2 (i.e. `ARROW_USER_SIMD_LEVEL=AVX) solved the issue

However, the table I use is only 3.8MB - so perhaps there is some other bug around the AVX2 related code here as well, unrelated to the size, but related to nulls.

Can unfortunately not share the table but if I figure a repro case I will drop it here.

Repro case:

import pyarrow as pa
def try_repro(size):
    repro = pa.table({"a": [0] * size,
                      "g": [None]*size},
                     schema=pa.schema([pa.field("a", "uint8"),
                                       pa.field("g", "date32")]))\
              .group_by(["a", "g"]).aggregate([([], "count_all")])

    if len(repro) != 1:
        print(f"{size} => {len(repro)}")
    return repro

for i in range(1,50):
    r = try_repro(i)

print()
print(r)

Output without AVX2 (expected):

$ ARROW_USER_SIMD_LEVEL=AVX python repro.py

pyarrow.Table
a: uint8
g: date32[day]
count_all: int64
----
a: [[0]]
g: [[null]]
count_all: [[49]]

Output with AVX2 (not expected):

ARROW_USER_SIMD_LEVEL=AVX2 python repro.py
33 => 2
...
40 => 2
41 => 3
...
48 => 3
49 => 4

pyarrow.Table
a: uint8
g: date32[day]
count_all: int64
----
a: [[0,0,0,0]]
g: [[null,null,null,null]]
count_all: [[32,8,8,1]]

Some observations:

  • Grouping on only g doesn't have the problem
  • Swapping the order a and g in the group_by also removes the issue.
  • Looks like this starts happening as soon as the size of the tables hits 33, and then we get an extra group for every 8 rows we add (so at 33, 41, 49)
  • Having g be an int does not exhibit the problem, a float does.
  • Non-null values don't have the issue
  • Macbook Pro M2 is also fine

Let me know if you think I should open a new ticket for this.

@amoeba
Copy link
Member

amoeba commented Jun 20, 2024

Hi @FreekPaans, can you please open a new issue for that? I think the issue will be fixed in the upcoming 17.x PyArrow release but it'd be good to make sure.

@FreekPaans
Copy link

@amoeba sure thing #42231

Any issue/PR you can point me to for where it's fixed?

@amoeba
Copy link
Member

amoeba commented Jun 20, 2024

Thanks. I'll test and follow up over on #42231.

@zanmato1984
Copy link
Contributor Author

@pitrou @felipecrv @ZhangHuiGui @mapleFU Would you please help to take a look? Thanks.

@@ -236,6 +236,8 @@ uint32_t KeyCompare::CompareBinaryColumnToRowHelper_avx2(
irow_right =
_mm256_loadu_si256(reinterpret_cast<const __m256i*>(left_to_right_map) + i);
}
// TODO: Need to test if this gather is OK when irow_right is larger than
Copy link
Contributor Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

I'll test in the future.

Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

When you say "in the future", is it in this PR or another one?

Copy link
Contributor Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Oh sorry, I meant in another PR.

@@ -236,6 +236,8 @@ uint32_t KeyCompare::CompareBinaryColumnToRowHelper_avx2(
irow_right =
_mm256_loadu_si256(reinterpret_cast<const __m256i*>(left_to_right_map) + i);
}
// TODO: Need to test if this gather is OK when irow_right is larger than
Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

When you say "in the future", is it in this PR or another one?

/// `0x80000000` lower. This way, the offset is always in range of [-2G, 2G) and those
/// intrinsics are safe.

constexpr auto two_gb = 0x80000000ull;
Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Can we make sure we use an explicit width type here? I'm not even sure what it is expected to be for correctness of the code using this constant (uint32_t or uint64_t?)

Copy link
Contributor Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Both uint32_t and uint64_t are OK. It only has to be unsigned and wide enough for 0x80000000. I'm declaring it uint64_t (the ull suffix) just to make all the arithmetics to be promoted to 64b to not worry about the potential underflow. The two subsequent usages are:

  1. Being added to pointer base after divided by a specific sizeof(). The division is unsigned so the addition is addressing the base "forward", as expected.
  2. Being loaded to a signed __m256i register via an implicit static cast (after divided by scale).

I'll update to make it, and the usages, more more type and width explicit.

Copy link
Contributor Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Addressed.

}

template <int scale>
inline __m256i UnsignedOffsetSafeGather64(arrow::util::int64_for_gather_t const* base,
Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

What is the use of int64_for_gather_t exactly?

Copy link
Contributor Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.


constexpr auto two_gb = 0x80000000ull;

template <int scale>
Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Two things:

  1. if we're using unsigned arithmetic below, the scale type should probably be unsigned for readability and sanity?
  2. naming convention: can we make this kScale?

Copy link
Contributor Author

@zanmato1984 zanmato1984 Jun 24, 2024

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

  1. The type of the third formal parameter of _mm256_set1_epi32/64 is int so I'm just using int too. Yeah, that's probably good.
  2. Yeah, will do.

Copy link
Contributor Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Addressed.

@@ -251,6 +253,35 @@ uint32_t KeyCompare::CompareBinaryColumnToRowHelper_avx2(
}
}

namespace {

/// Intrinsics `_mm256_i32gather_epi32/64` treat the `vindex` as signed integer, and we
Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Can you use regular comments (//)? This isn't a docstring so shouldn't use the docstring-specific prefix (///)

Copy link
Contributor Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Yeah, will do.

Copy link
Contributor Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Addressed.

// number of rows.
constexpr int64_t num_rows = std::numeric_limits<uint16_t>::max() + 1;
const std::vector<std::shared_ptr<DataType>> fixed_length_types{uint64(), uint32()};
// The var length column should be a little smaller than 2GB to WAR the capacity
Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

"WAR"?

Copy link
Contributor Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Sorry, I meant "workaround". Will update.

Copy link
Contributor Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Addressed.

@@ -164,5 +166,128 @@ TEST(KeyCompare, CompareColumnsToRowsTempStackUsage) {
}
}

// Compare columns to rows at offsets over 2GB within a row table.
// Certain AVX2 instructions may behave unexpectedly causing troubles like GH-41813.
TEST(KeyCompare, CompareColumnsToRowsLarge) {
Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

What is the runtime of this test? Perhaps we need to disable it on Valgrind builds.

Copy link
Contributor Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

What do you mean by "runtime"? I can't think of a reason why Valgrind would complain (at least ASAN didn't).

Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Sorry, I meant "run time" or execution time :-)

Copy link
Contributor Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Ah, got it! It takes about 20s with ASAN enabled. Perhaps it will be fine with Valgrind too?

Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

I should take a quick look.

Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Ok, it takes 70s locally under Valgrind. That's a bit high for a single test, I would rather disable it under Valgrind.

Copy link
Contributor Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

OK. Updated to disable the test under Valgrind. Thanks for helping running in your local!

ASSERT_OK(row_encoder.EncodeSelected(&row_table, static_cast<uint32_t>(num_rows),
row_ids_right.data()));

ASSERT_TRUE(row_table.offsets());
Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

I'm not sure what's that supposed to check (offsets being "true"?). Do we want to make the test a bit more self-documenting, or perhaps add a comment?

Copy link
Contributor Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

This is asserting the address of row_table.offsets() is not null, like if (some_pointer). Perhaps I can refine it to ASSERT_NE(row_table.offsets(), NULLPTR).

And the point of this check is to make sure the row_table constructed has an internal offset buffer, i.e., it contains var length columns.

Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Yes, the ASSERT_NE suggestion would make this more easily understandable, thanks!

Copy link
Contributor Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Addressed.

base + kTwoGB / sizeof(arrow::util::int64_for_gather_t);
__m128i normalized_offset =
_mm_sub_epi32(offset, _mm_set1_epi32(static_cast<int>(kTwoGB / kScale)));
return _mm256_i32gather_epi64(normalized_base, normalized_offset,
Copy link
Collaborator

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

I have a question about instructions.

Why is the vindex parameter type of _mm256_i32gather_epi32 is _m256i and the vindex type of _mm256_i32gather_epi64 is _m128i?

This may not be related to PR, I just want to understand it🫡

Copy link
Contributor Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Both intrinsics gather "several" integers based on a base address and "several" 32b offsets (vindex), and stores the results into a 256b register. The difference is: _mm256_i32gather_epi32 gathers 8 32b-integers (8 * 32 = 256) at a time so 8 32b indices are used, hence the 256b vindex. Whereas _mm256_i32gather_epi64 gathers 4 64b-integers at a time so 4 32b indices are used, hence the 128b vindex.

Copy link
Collaborator

@ZhangHuiGui ZhangHuiGui Jun 25, 2024

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

I see. Thanks!

@zanmato1984
Copy link
Contributor Author

zanmato1984 commented Jun 25, 2024

I've committed two changes containing code restructures (moving, renaming, etc.) and a minor fix to the test, to make the test logic more clear and readable. Hope it doesn't trouble your review @pitrou . Thanks.

@pitrou pitrou merged commit e635cc2 into apache:main Jun 25, 2024
35 of 39 checks passed
@pitrou pitrou removed the awaiting committer review Awaiting committer review label Jun 25, 2024
@github-actions github-actions bot added the awaiting committer review Awaiting committer review label Jun 25, 2024
@pitrou
Copy link
Member

pitrou commented Jun 25, 2024

Thanks a lot @zanmato1984 !

Copy link

After merging your PR, Conbench analyzed the 7 benchmarking runs that have been run so far on merge-commit e635cc2.

There were no benchmark performance regressions. 🎉

The full Conbench report has more details. It also includes information about 7 possible false positives for unstable benchmarks that are known to sometimes produce them.

pitrou pushed a commit that referenced this pull request Jul 10, 2024
…ColumnsToRows` (#43065)

### Rationale for this change

See #43046.

### What changes are included in this PR?

Use unsigned offset safe gather introduced in #42188 which is to fix similar issues.

### Are these changes tested?

Yes.

### Are there any user-facing changes?

None.

* GitHub Issue: #43046

Lead-authored-by: Ruoxi Sun <[email protected]>
Co-authored-by: Rossi Sun <[email protected]>
Signed-off-by: Antoine Pitrou <[email protected]>
Sign up for free to join this conversation on GitHub. Already have an account? Sign in to comment
Labels
awaiting committer review Awaiting committer review Component: C++ Critical Fix Bugfixes for security vulnerabilities, crashes, or invalid data.
Projects
None yet
Development

Successfully merging this pull request may close these issues.

6 participants