15f757f3fSDimitry Andric // -*- C++ -*-
25f757f3fSDimitry Andric //===----------------------------------------------------------------------===//
35f757f3fSDimitry Andric //
45f757f3fSDimitry Andric // Part of the LLVM Project, under the Apache License v2.0 with LLVM Exceptions.
55f757f3fSDimitry Andric // See https://llvm.org/LICENSE.txt for license information.
65f757f3fSDimitry Andric // SPDX-License-Identifier: Apache-2.0 WITH LLVM-exception
75f757f3fSDimitry Andric //
85f757f3fSDimitry Andric //===----------------------------------------------------------------------===//
95f757f3fSDimitry Andric
105f757f3fSDimitry Andric #ifndef _LIBCPP_EXPERIMENTAL___SIMD_REFERENCE_H
115f757f3fSDimitry Andric #define _LIBCPP_EXPERIMENTAL___SIMD_REFERENCE_H
125f757f3fSDimitry Andric
135f757f3fSDimitry Andric #include <__type_traits/is_assignable.h>
145f757f3fSDimitry Andric #include <__type_traits/is_same.h>
155f757f3fSDimitry Andric #include <__utility/forward.h>
16*0fca6ea1SDimitry Andric #include <__utility/move.h>
175f757f3fSDimitry Andric #include <cstddef>
185f757f3fSDimitry Andric #include <experimental/__config>
195f757f3fSDimitry Andric #include <experimental/__simd/utility.h>
205f757f3fSDimitry Andric
21*0fca6ea1SDimitry Andric _LIBCPP_PUSH_MACROS
22*0fca6ea1SDimitry Andric #include <__undef_macros>
23*0fca6ea1SDimitry Andric
245f757f3fSDimitry Andric #if _LIBCPP_STD_VER >= 17 && defined(_LIBCPP_ENABLE_EXPERIMENTAL)
255f757f3fSDimitry Andric
265f757f3fSDimitry Andric _LIBCPP_BEGIN_NAMESPACE_EXPERIMENTAL
275f757f3fSDimitry Andric inline namespace parallelism_v2 {
285f757f3fSDimitry Andric template <class _Tp, class _Storage, class _Vp>
295f757f3fSDimitry Andric class __simd_reference {
305f757f3fSDimitry Andric template <class, class>
315f757f3fSDimitry Andric friend class simd;
325f757f3fSDimitry Andric template <class, class>
335f757f3fSDimitry Andric friend class simd_mask;
345f757f3fSDimitry Andric
355f757f3fSDimitry Andric _Storage& __s_;
365f757f3fSDimitry Andric size_t __idx_;
375f757f3fSDimitry Andric
__simd_reference(_Storage & __s,size_t __idx)385f757f3fSDimitry Andric _LIBCPP_HIDE_FROM_ABI __simd_reference(_Storage& __s, size_t __idx) : __s_(__s), __idx_(__idx) {}
395f757f3fSDimitry Andric
__get()405f757f3fSDimitry Andric _LIBCPP_HIDE_FROM_ABI _Vp __get() const noexcept { return __s_.__get(__idx_); }
415f757f3fSDimitry Andric
__set(_Vp __v)425f757f3fSDimitry Andric _LIBCPP_HIDE_FROM_ABI void __set(_Vp __v) {
435f757f3fSDimitry Andric if constexpr (is_same_v<_Vp, bool>)
445f757f3fSDimitry Andric __s_.__set(__idx_, experimental::__set_all_bits<_Tp>(__v));
455f757f3fSDimitry Andric else
465f757f3fSDimitry Andric __s_.__set(__idx_, __v);
475f757f3fSDimitry Andric }
485f757f3fSDimitry Andric
495f757f3fSDimitry Andric public:
505f757f3fSDimitry Andric using value_type = _Vp;
515f757f3fSDimitry Andric
525f757f3fSDimitry Andric __simd_reference() = delete;
535f757f3fSDimitry Andric __simd_reference(const __simd_reference&) = delete;
545f757f3fSDimitry Andric
value_type()555f757f3fSDimitry Andric _LIBCPP_HIDE_FROM_ABI operator value_type() const noexcept { return __get(); }
565f757f3fSDimitry Andric
575f757f3fSDimitry Andric template <class _Up, enable_if_t<is_assignable_v<value_type&, _Up&&>, int> = 0>
585f757f3fSDimitry Andric _LIBCPP_HIDE_FROM_ABI __simd_reference operator=(_Up&& __v) && noexcept {
595f757f3fSDimitry Andric __set(static_cast<value_type>(std::forward<_Up>(__v)));
605f757f3fSDimitry Andric return {__s_, __idx_};
615f757f3fSDimitry Andric }
62*0fca6ea1SDimitry Andric
63*0fca6ea1SDimitry Andric // Note: This approach might not fully align with the specification,
64*0fca6ea1SDimitry Andric // which might be a wording defect. (https://wg21.link/N4808 section 9.6.3)
65*0fca6ea1SDimitry Andric template <class _Tp1, class _Storage1, class _Vp1>
66*0fca6ea1SDimitry Andric friend void
67*0fca6ea1SDimitry Andric swap(__simd_reference<_Tp1, _Storage1, _Vp1>&& __a, __simd_reference<_Tp1, _Storage1, _Vp1>&& __b) noexcept;
68*0fca6ea1SDimitry Andric
69*0fca6ea1SDimitry Andric template <class _Tp1, class _Storage1, class _Vp1>
70*0fca6ea1SDimitry Andric friend void swap(_Vp1& __a, __simd_reference<_Tp1, _Storage1, _Vp1>&& __b) noexcept;
71*0fca6ea1SDimitry Andric
72*0fca6ea1SDimitry Andric template <class _Tp1, class _Storage1, class _Vp1>
73*0fca6ea1SDimitry Andric friend void swap(__simd_reference<_Tp1, _Storage1, _Vp1>&& __a, _Vp1& __b) noexcept;
745f757f3fSDimitry Andric };
755f757f3fSDimitry Andric
76*0fca6ea1SDimitry Andric template <class _Tp, class _Storage, class _Vp>
77*0fca6ea1SDimitry Andric _LIBCPP_HIDE_FROM_ABI void
swap(__simd_reference<_Tp,_Storage,_Vp> && __a,__simd_reference<_Tp,_Storage,_Vp> && __b)78*0fca6ea1SDimitry Andric swap(__simd_reference<_Tp, _Storage, _Vp>&& __a, __simd_reference<_Tp, _Storage, _Vp>&& __b) noexcept {
79*0fca6ea1SDimitry Andric _Vp __tmp(std::move(__a));
80*0fca6ea1SDimitry Andric std::move(__a) = std::move(__b);
81*0fca6ea1SDimitry Andric std::move(__b) = std::move(__tmp);
82*0fca6ea1SDimitry Andric }
83*0fca6ea1SDimitry Andric
84*0fca6ea1SDimitry Andric template <class _Tp, class _Storage, class _Vp>
swap(_Vp & __a,__simd_reference<_Tp,_Storage,_Vp> && __b)85*0fca6ea1SDimitry Andric _LIBCPP_HIDE_FROM_ABI void swap(_Vp& __a, __simd_reference<_Tp, _Storage, _Vp>&& __b) noexcept {
86*0fca6ea1SDimitry Andric _Vp __tmp(std::move(__a));
87*0fca6ea1SDimitry Andric __a = std::move(__b);
88*0fca6ea1SDimitry Andric std::move(__b) = std::move(__tmp);
89*0fca6ea1SDimitry Andric }
90*0fca6ea1SDimitry Andric
91*0fca6ea1SDimitry Andric template <class _Tp, class _Storage, class _Vp>
swap(__simd_reference<_Tp,_Storage,_Vp> && __a,_Vp & __b)92*0fca6ea1SDimitry Andric _LIBCPP_HIDE_FROM_ABI void swap(__simd_reference<_Tp, _Storage, _Vp>&& __a, _Vp& __b) noexcept {
93*0fca6ea1SDimitry Andric _Vp __tmp(std::move(__a));
94*0fca6ea1SDimitry Andric std::move(__a) = std::move(__b);
95*0fca6ea1SDimitry Andric __b = std::move(__tmp);
96*0fca6ea1SDimitry Andric }
97*0fca6ea1SDimitry Andric
985f757f3fSDimitry Andric } // namespace parallelism_v2
995f757f3fSDimitry Andric _LIBCPP_END_NAMESPACE_EXPERIMENTAL
1005f757f3fSDimitry Andric
1015f757f3fSDimitry Andric #endif // _LIBCPP_STD_VER >= 17 && defined(_LIBCPP_ENABLE_EXPERIMENTAL)
102*0fca6ea1SDimitry Andric
103*0fca6ea1SDimitry Andric _LIBCPP_POP_MACROS
104*0fca6ea1SDimitry Andric
1055f757f3fSDimitry Andric #endif // _LIBCPP_EXPERIMENTAL___SIMD_REFERENCE_H
106