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

Add workaround to parallel merge in TBB backend not to use input iterators default constructor #1346

Open
wants to merge 3 commits into
base: main
Choose a base branch
from

Conversation

kboyarinov
Copy link
Contributor

@kboyarinov kboyarinov commented Jan 15, 2024

Add workaround to implementation of __parallel_merge function for the following use case:

#include <oneapi/dpl/algorithm>
#include <oneapi/dpl/execution>
#include <oneapi/dpl/iterator>
#include <vector>

int main() {
    std::vector<int> input1 = {1, 2, 3};
    std::vector<int> input2 = {1, 2, 3};
    std::vector<int> output = {0, 0, 0};

    auto no_transformation = [](int i) { return i; };

    oneapi::dpl::merge(oneapi::dpl::execution::par,
                       oneapi::dpl::make_transform_iterator(input1.begin(), no_transformation),
                       oneapi::dpl::make_transform_iterator(input1.end(), no_transformation),
                       input2.begin(),
                       input2.end(),
                       output.begin());
}

The first input iterator in merge algorithm is dpl::transform_iterator<vector_it, lambda>. Since lambda is not default constructible, this use case results in the compilation error while trying to default construct iterator inside TBB backend.

From the C++ Standard perspective, it is not an issue, because default constructability is part of ForwardIterator requirements, but after the short offline discussion, it was decided to workaround this place in parallel backend TBB to make things better for our fancy iterators.

There is still no 100% guarantee, that no issues would be present for transform iterator with lambda body, because some algorithms we use from the C++ Standard Library are still allowed to call default constructor on the input iterator, e.g. std::upper_bound). This PR was tested together with libstdc++ and no issues was found.

@SergeyKopienko
Copy link
Contributor

SergeyKopienko commented Jan 15, 2024

I think we are able to fix this issue without changes in transform_iterator and it's more preferable: for that we should duplicate some code in if / else parts inside https://github.com/oneapi-src/oneDPL/pull/1346/files#diff-9908a4470cd112b8a7c7c87c5d3dfa425d873a5e5f9bf353268e6bdf756f5af8.

@SergeyKopienko
Copy link
Contributor

What about the code

        // Using callable object instead of lambda here to ensure transform iterator would be
        // default constructible, that is part of the Forward Iterator requirements in the C++ standard.
        struct NoTransform
        {
            ValueType operator()(const ValueType& val) const
            {
                return val;
            }
        };

Should we use lambda now in https://github.com/oneapi-src/oneDPL/blob/5740f585386bed798af83e1c18ae1ee92c613637/test/parallel_api/iterator/permutation_iterator.h#L211 ?

@kboyarinov
Copy link
Contributor Author

What about the code

        // Using callable object instead of lambda here to ensure transform iterator would be
        // default constructible, that is part of the Forward Iterator requirements in the C++ standard.
        struct NoTransform
        {
            ValueType operator()(const ValueType& val) const
            {
                return val;
            }
        };

Should we use lambda now in

https://github.com/oneapi-src/oneDPL/blob/5740f585386bed798af83e1c18ae1ee92c613637/test/parallel_api/iterator/permutation_iterator.h#L211

?

I would prefer to keep this test as-is because, as I mentioned this fix is not a 100% guarantee that merge with transform_iterator with lambda body would work. std::upper_bound and std::lower_bound implementations can still rely on default constructability of input iterators.
For today, this code works, but formally such an iterator violates the ForwardIterator requirements from the Standard.
This patch is intended to minimize the risk of issues, as we discussed offline.

@SergeyKopienko
Copy link
Contributor

What about the code

        // Using callable object instead of lambda here to ensure transform iterator would be
        // default constructible, that is part of the Forward Iterator requirements in the C++ standard.
        struct NoTransform
        {
            ValueType operator()(const ValueType& val) const
            {
                return val;
            }
        };

Should we use lambda now in
https://github.com/oneapi-src/oneDPL/blob/5740f585386bed798af83e1c18ae1ee92c613637/test/parallel_api/iterator/permutation_iterator.h#L211

?

I would prefer to keep this test as-is because, as I mentioned this fix is not a 100% guarantee that merge with transform_iterator with lambda body would work. std::upper_bound and std::lower_bound implementations can still rely on default constructability of input iterators. For today, this code works, but formally such an iterator violates the ForwardIterator requirements from the Standard. This patch is intended to minimize the risk of issues, as we discussed offline.

ok

Copy link
Contributor

@SergeyKopienko SergeyKopienko left a comment

Choose a reason for hiding this comment

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

LGTM

@akukanov
Copy link
Contributor

Should the patch also be done in PSTL upstream?
@MikeDvorskiy @rarutyun

__ym = __ys + (__size_y / 2);
__xm = std::upper_bound(__xs, __xe, *__ym, __comp);
_RandomAccessIterator2 __ym = __ys + (__size_y / 2);
__make_parallel_merge_leaf_tasks(::std::upper_bound(__xs, __xe, *__ym, __comp),
Copy link
Contributor

Choose a reason for hiding this comment

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

Recently, we came to conclusion that we should not write :: operator before std:: in new code, at least.

@MikeDvorskiy
Copy link
Contributor

Should the patch also be done in PSTL upstream? @MikeDvorskiy @rarutyun

I guess, yes it should.

@akukanov akukanov added the follow through PRs/issues that should be completed/resolved label Oct 18, 2024
Sign up for free to join this conversation on GitHub. Already have an account? Sign in to comment
Labels
follow through PRs/issues that should be completed/resolved
Projects
None yet
Development

Successfully merging this pull request may close these issues.

4 participants