162 lines
		
	
	
		
			6.4 KiB
		
	
	
	
		
			C++
		
	
	
	
	
	
			
		
		
	
	
			162 lines
		
	
	
		
			6.4 KiB
		
	
	
	
		
			C++
		
	
	
	
	
	
| /***************************************************************************************************
 | |
|  * Copyright (c) 2023 - 2023 NVIDIA CORPORATION & AFFILIATES. All rights reserved.
 | |
|  * SPDX-License-Identifier: BSD-3-Clause
 | |
|  *
 | |
|  * Redistribution and use in source and binary forms, with or without
 | |
|  * modification, are permitted provided that the following conditions are met:
 | |
|  *
 | |
|  * 1. Redistributions of source code must retain the above copyright notice, this
 | |
|  * list of conditions and the following disclaimer.
 | |
|  *
 | |
|  * 2. Redistributions in binary form must reproduce the above copyright notice,
 | |
|  * this list of conditions and the following disclaimer in the documentation
 | |
|  * and/or other materials provided with the distribution.
 | |
|  *
 | |
|  * 3. Neither the name of the copyright holder nor the names of its
 | |
|  * contributors may be used to endorse or promote products derived from
 | |
|  * this software without specific prior written permission.
 | |
|  *
 | |
|  * THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "AS IS"
 | |
|  * AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
 | |
|  * IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE
 | |
|  * DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT HOLDER OR CONTRIBUTORS BE LIABLE
 | |
|  * FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL
 | |
|  * DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR
 | |
|  * SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER
 | |
|  * CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY,
 | |
|  * OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE
 | |
|  * OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
 | |
|  *
 | |
|  **************************************************************************************************/
 | |
| 
 | |
| #include "cutlass_unit_test.h"
 | |
| 
 | |
| #include <cutlass/trace.h>
 | |
| 
 | |
| #include <cassert>
 | |
| #include <type_traits>
 | |
| 
 | |
| #include <cute/container/tuple.hpp>
 | |
| #include <cute/int_tuple.hpp>
 | |
| 
 | |
| template<class T>
 | |
| class ConvertibleTo {
 | |
| public:
 | |
|   ConvertibleTo(T val) : val_(val) {}
 | |
| 
 | |
|   operator T () const { return val_; }
 | |
| 
 | |
| private:
 | |
|   T val_ = 0;
 | |
| };
 | |
| 
 | |
| template<class Integral, Integral Value>
 | |
| using IC = std::integral_constant<Integral, Value>;
 | |
| 
 | |
| TEST(CuTe_core_msvc_compilation, TupleAssignment)
 | |
| {
 | |
|   CUTLASS_TRACE_HOST("-------------------------------");
 | |
|   CUTLASS_TRACE_HOST("cute::tuple creation and assignment");
 | |
|   CUTLASS_TRACE_HOST("-------------------------------");
 | |
| 
 | |
|   using forty_two_type = IC<int, 42>;
 | |
|   using forty_three_type = IC<size_t, 43>;
 | |
| 
 | |
|   using ebo_s_type = cute::detail::EBO<0, forty_two_type>;
 | |
|   [[maybe_unused]] ebo_s_type ebo_s;
 | |
|   static_assert(std::is_same_v<decltype(cute::detail::getv(ebo_s)), forty_two_type>);
 | |
| 
 | |
|   using ebo_d_type = cute::detail::EBO<1, size_t>;
 | |
|   [[maybe_unused]] ebo_d_type ebo_d(43u);
 | |
|   assert(ebo_d.t_ == 43u);
 | |
|   static_assert(std::is_same_v<std::remove_const_t<std::remove_reference_t<decltype(cute::detail::getv(ebo_d))>>, size_t > );
 | |
|   assert(cute::detail::getv(ebo_d) == 43u);
 | |
| 
 | |
|   [[maybe_unused]] cute::detail::TupleBase<std::index_sequence<0, 1, 2>, int, forty_two_type, size_t> tb0{
 | |
|           41, forty_two_type{}, size_t(43u) };
 | |
|   [[maybe_unused]] cute::detail::TupleBase<std::index_sequence<0, 1, 2>, int, forty_two_type, size_t> tb1;
 | |
| 
 | |
|   int val41 = ConvertibleTo{41};
 | |
|   assert(val41 == 41);
 | |
|   size_t val43 = ConvertibleTo{size_t(43u)};
 | |
|   assert(val43 == size_t{43u});
 | |
|   [[maybe_unused]] cute::detail::TupleBase<std::index_sequence<0, 1, 2>, int, forty_two_type, size_t> tb2{
 | |
|         ConvertibleTo{41}, forty_two_type{}, ConvertibleTo{size_t(43u)}};
 | |
| 
 | |
|   [[maybe_unused]] cute::detail::TupleBase<std::index_sequence<0>, int> tb3{ 41 };
 | |
|   [[maybe_unused]] cute::detail::TupleBase<std::index_sequence<0>, int> tb3a{ 42 };
 | |
|   tb3 = tb3a;
 | |
| 
 | |
|   using tuple_0d_type = cute::tuple<>;
 | |
|   using tuple_1d_d_type = cute::tuple<int>;
 | |
|   using tuple_1d_s_type = cute::tuple<forty_two_type>;
 | |
|   using tuple_2d_dd_type = cute::tuple<int, size_t>;
 | |
|   using tuple_2d_ss_type = cute::tuple<forty_two_type, forty_three_type>;
 | |
| 
 | |
|   [[maybe_unused]] tuple_0d_type t0;
 | |
| 
 | |
|   // Symptom: "illegal member initialization: 'TupleBase<int>' is not a base or member"
 | |
|   [[maybe_unused]] tuple_1d_d_type t1{ 42 };
 | |
| 
 | |
|   [[maybe_unused]] tuple_1d_s_type t2;
 | |
| 
 | |
|   [[maybe_unused]] tuple_1d_d_type t1a{ 43 };
 | |
|   t1 = t1a;
 | |
| 
 | |
|   [[maybe_unused]] tuple_2d_dd_type t3{ 42, size_t(43u) };
 | |
|   [[maybe_unused]] tuple_2d_ss_type t4;
 | |
|   t3 = t4;
 | |
| 
 | |
|   [[maybe_unused]] tuple_2d_dd_type t3a{ 44, size_t(45u) };
 | |
|   // Symptom: "illegal member initialization:
 | |
|   // 'TupleBase<int, unsigned __int64>' is not a base or member"
 | |
|   t3 = t3a;
 | |
| }
 | |
| 
 | |
| TEST(CuTe_core_msvc_compilation, TupleGetSingleInteger)
 | |
| {
 | |
|   CUTLASS_TRACE_HOST("-------------------------------");
 | |
|   CUTLASS_TRACE_HOST("cute::get<I> on cute::tuple for single integer I");
 | |
|   CUTLASS_TRACE_HOST("-------------------------------");
 | |
| 
 | |
|   cute::tuple<int, ConvertibleTo<size_t>, IC<int, 43>> t0{ 41, size_t(42u), IC<int, 43>{} };
 | |
| 
 | |
|   [[maybe_unused]] auto t0_0 = cute::get<0>(t0);
 | |
|   static_assert(std::is_same_v<decltype(t0_0), int>);
 | |
|   assert(t0_0 == 41);
 | |
| 
 | |
|   [[maybe_unused]] auto t0_1 = cute::get<1>(t0);
 | |
|   static_assert(std::is_same_v<decltype(t0_1), ConvertibleTo<size_t>>);
 | |
| 
 | |
|   [[maybe_unused]] auto t0_2 = cute::get<2>(t0);
 | |
|   static_assert(std::is_same_v<decltype(t0_2), IC<int, 43>>);
 | |
| }
 | |
| 
 | |
| TEST(CuTe_core_msvc_compilation, TupleGetRecursive)
 | |
| {
 | |
|   CUTLASS_TRACE_HOST("-------------------------------");
 | |
|   CUTLASS_TRACE_HOST("cute::get<I...> on cute::tuple");
 | |
|   CUTLASS_TRACE_HOST("-------------------------------");
 | |
| 
 | |
|   using inner_tuple_type = cute::tuple<int, ConvertibleTo<size_t>, IC<int, 43>>;
 | |
|   using outer_tuple_type = cute::tuple<IC<int, 40>, inner_tuple_type, size_t>;
 | |
| 
 | |
|   inner_tuple_type t0_inner{ 41, size_t(42u), IC<int, 43>{} };
 | |
|   outer_tuple_type t0_outer{ IC<int, 40>{}, t0_inner, size_t(44u) };
 | |
| 
 | |
|         [[maybe_unused]] auto t0_outer_0 = cute::get<0>(t0_outer);
 | |
|         static_assert(std::is_same_v<decltype(t0_outer_0), IC<int, 40>>);
 | |
| 
 | |
|         [[maybe_unused]] auto t0_outer_1 = cute::get<1>(t0_outer);
 | |
|         static_assert(std::is_same_v<decltype(t0_outer_1), inner_tuple_type>);
 | |
| 
 | |
|         [[maybe_unused]] auto t0_outer_2 = cute::get<2>(t0_outer);
 | |
|         static_assert(std::is_same_v<decltype(t0_outer_2), size_t>);
 | |
|         assert(t0_outer_2 == size_t(44u));
 | |
| 
 | |
|   // Leftmost index is innermost in the nexted get sequence.
 | |
|   [[maybe_unused]] auto t0_outer_10 = cute::get<1, 0>(t0_outer);
 | |
|   static_assert(std::is_same_v<decltype(t0_outer_10), int>);
 | |
|   assert(t0_outer_10 == 41);
 | |
| }
 | 
