| 
									
										
										
										
											2021-07-23 12:40:53 +08:00
										 |  |  | /***************************************************************************************************
 | 
					
						
							| 
									
										
										
										
											2022-04-24 03:02:38 +08:00
										 |  |  |  * Copyright (c) 2017 - 2022 NVIDIA CORPORATION & AFFILIATES. All rights reserved. | 
					
						
							|  |  |  |  * SPDX-License-Identifier: BSD-3-Clause | 
					
						
							| 
									
										
										
										
											2021-07-23 12:40:53 +08:00
										 |  |  |  * | 
					
						
							| 
									
										
										
										
											2022-04-24 03:02:38 +08:00
										 |  |  |  * Redistribution and use in source and binary forms, with or without | 
					
						
							|  |  |  |  * modification, are permitted provided that the following conditions are met: | 
					
						
							| 
									
										
										
										
											2021-07-23 12:40:53 +08:00
										 |  |  |  * | 
					
						
							| 
									
										
										
										
											2022-04-24 03:02:38 +08:00
										 |  |  |  * 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 | 
					
						
							| 
									
										
										
										
											2021-07-23 12:40:53 +08:00
										 |  |  |  * OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE. | 
					
						
							|  |  |  |  * | 
					
						
							|  |  |  |  **************************************************************************************************/ | 
					
						
							|  |  |  | /*! \file
 | 
					
						
							|  |  |  |     \brief Implicit GEMM testbed | 
					
						
							|  |  |  | */ | 
					
						
							|  |  |  | #pragma once
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-07-20 03:23:54 +08:00
										 |  |  | #include <fstream>
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-07-23 12:40:53 +08:00
										 |  |  | #include "../../common/cutlass_unit_test.h"
 | 
					
						
							|  |  |  | #include "cutlass/cutlass.h"
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | #include "cutlass/conv/device/implicit_gemm_convolution.h"
 | 
					
						
							| 
									
										
										
										
											2022-05-07 01:24:51 +08:00
										 |  |  | #include "cutlass/reduction/device/tensor_reduce.h"
 | 
					
						
							| 
									
										
										
										
											2021-07-23 12:40:53 +08:00
										 |  |  | #include "cutlass/reduction/device/reduce_split_k.h"
 | 
					
						
							|  |  |  | #include "cutlass/reduction/thread/reduction_operators.h"
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | #include "conv2d_problems.h"
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | #include "cutlass/util/host_tensor.h"
 | 
					
						
							|  |  |  | #include "cutlass/util/reference/host/tensor_fill.h"
 | 
					
						
							|  |  |  | #include "cutlass/util/reference/device/tensor_compare.h"
 | 
					
						
							|  |  |  | #include "cutlass/util/reference/host/tensor_compare.h"
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | #include "cutlass/util/reference/host/convolution.h"
 | 
					
						
							|  |  |  | #include "cutlass/util/reference/device/convolution.h"
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | #include "cutlass/core_io.h"
 | 
					
						
							|  |  |  | #include "cutlass/util/tensor_view_io.h"
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-09-21 02:02:22 +08:00
										 |  |  | #include "cache_testbed_output.h"
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-07-23 12:40:53 +08:00
										 |  |  | namespace test { | 
					
						
							|  |  |  | namespace conv { | 
					
						
							|  |  |  | namespace device { | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | template <typename Conv2d> | 
					
						
							|  |  |  | class TestbedConv2dWithReduction { | 
					
						
							|  |  |  | public: | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   using ElementA = typename Conv2d::ElementA; | 
					
						
							|  |  |  |   using LayoutA = typename Conv2d::LayoutA; | 
					
						
							|  |  |  |   using ElementB = typename Conv2d::ElementB; | 
					
						
							|  |  |  |   using LayoutB = typename Conv2d::LayoutB; | 
					
						
							|  |  |  |   using ElementC = typename Conv2d::ElementC; | 
					
						
							|  |  |  |   using LayoutC = typename Conv2d::LayoutC; | 
					
						
							|  |  |  |   using ElementAccumulator = typename Conv2d::ElementAccumulator; | 
					
						
							|  |  |  |   using ElementCompute = typename Conv2d::ElementCompute; | 
					
						
							|  |  |  |   using EpilogueOutputOp = typename Conv2d::EpilogueOutputOp; | 
					
						
							|  |  |  |   using ElementT = typename EpilogueOutputOp::ElementTensor; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   static cutlass::conv::Operator const kConvolutionalOperator = Conv2d::kConvolutionalOperator; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | public: | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   /// Initialization
 | 
					
						
							|  |  |  |   cutlass::Distribution::Kind init_A; | 
					
						
							|  |  |  |   cutlass::Distribution::Kind init_B; | 
					
						
							|  |  |  |   cutlass::Distribution::Kind init_C; | 
					
						
							|  |  |  |   uint64_t seed; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   cutlass::HostTensor<ElementA, LayoutA> tensor_A; | 
					
						
							|  |  |  |   cutlass::HostTensor<ElementB, LayoutB> tensor_B; | 
					
						
							|  |  |  |   cutlass::HostTensor<ElementC, LayoutC> tensor_C; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-05-07 01:24:51 +08:00
										 |  |  |   cutlass::HostTensor<ElementAccumulator, LayoutC> tensor_Reduction; | 
					
						
							| 
									
										
										
										
											2021-07-23 12:40:53 +08:00
										 |  |  |   cutlass::HostTensor<ElementT,           cutlass::layout::RowMajor> tensor_Tensor; | 
					
						
							| 
									
										
										
										
											2022-05-07 01:24:51 +08:00
										 |  |  |   cutlass::HostTensor<ElementAccumulator, LayoutC> tensor_Final_Reduction; | 
					
						
							| 
									
										
										
										
											2021-07-23 12:40:53 +08:00
										 |  |  | 
 | 
					
						
							|  |  |  |   cutlass::HostTensor<ElementC, LayoutC> tensor_D_computed; | 
					
						
							|  |  |  |   cutlass::HostTensor<ElementC, LayoutC> tensor_D_reference; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | public: | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   TestbedConv2dWithReduction( | 
					
						
							|  |  |  |     cutlass::Distribution::Kind init_A_ = cutlass::Distribution::Uniform, | 
					
						
							|  |  |  |     cutlass::Distribution::Kind init_B_ = cutlass::Distribution::Uniform, | 
					
						
							|  |  |  |     cutlass::Distribution::Kind init_C_ = cutlass::Distribution::Uniform, | 
					
						
							|  |  |  |     uint64_t seed_ = 2080 | 
					
						
							|  |  |  |   ): | 
					
						
							|  |  |  |     init_A(init_A_), init_B(init_B_), init_C(init_C_), seed(seed_) { | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     /// Helper to initialize a tensor view
 | 
					
						
							|  |  |  |   template <typename Element, typename Layout> | 
					
						
							|  |  |  |   void initialize_tensor( | 
					
						
							|  |  |  |     cutlass::TensorView<Element, Layout> view,  | 
					
						
							|  |  |  |     cutlass::Distribution::Kind dist_kind, | 
					
						
							|  |  |  |     uint64_t seed) { | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     if (dist_kind == cutlass::Distribution::Uniform) { | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-05-07 01:24:51 +08:00
										 |  |  |       int scope = 2; | 
					
						
							| 
									
										
										
										
											2021-07-23 12:40:53 +08:00
										 |  |  | 
 | 
					
						
							|  |  |  |       cutlass::reference::host::TensorFillRandomUniform( | 
					
						
							|  |  |  |         view, seed, scope, -scope, 0); | 
					
						
							|  |  |  |     }  | 
					
						
							|  |  |  |     else if (dist_kind == cutlass::Distribution::Identity) { | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |       cutlass::reference::host::TensorFillIdentity(view); | 
					
						
							|  |  |  |     }  | 
					
						
							|  |  |  |     else if (dist_kind == cutlass::Distribution::Gaussian) { | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |       cutlass::reference::host::TensorFillRandomGaussian(view, seed, 0, 0.5); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     else if (dist_kind == cutlass::Distribution::Sequential) { | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |       cutlass::reference::host::BlockFillSequential(view.data(), view.capacity()); | 
					
						
							|  |  |  |     }  | 
					
						
							|  |  |  |     else { | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   void initialize( | 
					
						
							|  |  |  |     cutlass::conv::Conv2dProblemSize const &problem_size, uint64_t seed = 2019) { | 
					
						
							|  |  |  |          | 
					
						
							|  |  |  |     tensor_A.resize(implicit_gemm_tensor_a_extent(kConvolutionalOperator, problem_size)); | 
					
						
							|  |  |  |     tensor_B.resize(implicit_gemm_tensor_b_extent(kConvolutionalOperator, problem_size)); | 
					
						
							|  |  |  |     tensor_C.resize(implicit_gemm_tensor_c_extent(kConvolutionalOperator, problem_size)); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     tensor_Reduction.resize({ | 
					
						
							| 
									
										
										
										
											2022-05-07 01:24:51 +08:00
										 |  |  |       1, | 
					
						
							|  |  |  |       1, | 
					
						
							|  |  |  |       (problem_size.N * problem_size.P * problem_size.Q - 1 + Conv2d::ThreadblockShape::kM) / Conv2d::ThreadblockShape::kM, | 
					
						
							|  |  |  |       (problem_size.K) | 
					
						
							|  |  |  |     }); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     tensor_Final_Reduction.resize({ | 
					
						
							|  |  |  |       1, | 
					
						
							|  |  |  |       1, | 
					
						
							|  |  |  |       1, | 
					
						
							|  |  |  |       (problem_size.K) | 
					
						
							| 
									
										
										
										
											2021-07-23 12:40:53 +08:00
										 |  |  |     }); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     tensor_Tensor.resize({(problem_size.N * problem_size.P * problem_size.Q), problem_size.K}); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     tensor_D_computed.resize(implicit_gemm_tensor_c_extent(kConvolutionalOperator, problem_size)); | 
					
						
							|  |  |  |     tensor_D_reference.resize(implicit_gemm_tensor_c_extent(kConvolutionalOperator, problem_size)); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     initialize_tensor(tensor_A.host_view(), init_A, seed);  | 
					
						
							|  |  |  |     initialize_tensor(tensor_B.host_view(), init_B, seed * 17);  | 
					
						
							|  |  |  |     initialize_tensor(tensor_C.host_view(), init_C, seed * 39); | 
					
						
							|  |  |  |      | 
					
						
							|  |  |  |     tensor_A.sync_device(); | 
					
						
							|  |  |  |     tensor_B.sync_device(); | 
					
						
							|  |  |  |     tensor_C.sync_device(); | 
					
						
							|  |  |  |     tensor_D_computed.sync_device(); | 
					
						
							|  |  |  |     tensor_D_reference.sync_device(); | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   bool sufficient() const { | 
					
						
							|  |  |  |     //
 | 
					
						
							|  |  |  |     // Determine SMEM requirements and waive if not satisfied
 | 
					
						
							|  |  |  |     //
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     int smem_size = int(sizeof(typename Conv2d::ImplicitGemmKernel::SharedStorage)); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     cudaDeviceProp properties; | 
					
						
							|  |  |  |     int device_idx; | 
					
						
							|  |  |  |     cudaError_t result = cudaGetDevice(&device_idx); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     if (result != cudaSuccess) { | 
					
						
							|  |  |  |       throw std::runtime_error("cudaGetDevice() API call failed."); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     result = cudaGetDeviceProperties(&properties, device_idx); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     if (result != cudaSuccess) { | 
					
						
							|  |  |  |       throw std::runtime_error("cudaGetDeviceProperties() failed"); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     if (properties.sharedMemPerMultiprocessor < smem_size) { | 
					
						
							|  |  |  |       return false; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     return true; | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   /// Executes one test
 | 
					
						
							|  |  |  |   bool run( | 
					
						
							|  |  |  |     cutlass::conv::Conv2dProblemSize const &problem_size, | 
					
						
							|  |  |  |     cutlass::conv::SplitKMode const &split_k_mode = cutlass::conv::SplitKMode::kSerial, | 
					
						
							|  |  |  |     ElementCompute alpha = ElementCompute(1), | 
					
						
							|  |  |  |     ElementCompute beta = ElementCompute(0)) { | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     // Waive test if insufficient CUDA device
 | 
					
						
							|  |  |  |     if (!sufficient()) { | 
					
						
							|  |  |  |       if (CUTLASS_TEST_UNIT_ENABLE_WARNINGS) { | 
					
						
							|  |  |  |         std::cerr << "Test waived due to insufficient CUDA device." << std::endl; | 
					
						
							|  |  |  |       } | 
					
						
							|  |  |  |       return true; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | #if 0 //display conv2d problem size for debugging
 | 
					
						
							|  |  |  |     std::cout << problem_size << std::endl | 
					
						
							|  |  |  |               << "alpha, beta: (" << alpha << ", " << beta << ")" << std::endl | 
					
						
							|  |  |  |               << "split_k_mode: " << ((split_k_mode == cutlass::conv::SplitKMode::kSerial) ? "(serial)" : "(parallel)") << std::endl | 
					
						
							|  |  |  |               << std::endl; | 
					
						
							|  |  |  | #endif
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     initialize(problem_size); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     // configure the operator
 | 
					
						
							|  |  |  |     Conv2d conv2d_op; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     typename Conv2d::Arguments conv2d_args( | 
					
						
							|  |  |  |       problem_size, | 
					
						
							|  |  |  |       tensor_A.device_ref(), | 
					
						
							|  |  |  |       tensor_B.device_ref(), | 
					
						
							|  |  |  |       tensor_C.device_ref(), | 
					
						
							|  |  |  |       tensor_D_computed.device_ref(), | 
					
						
							|  |  |  |       {alpha, beta}, | 
					
						
							|  |  |  |       split_k_mode, | 
					
						
							|  |  |  |       tensor_Reduction.device_data(), | 
					
						
							|  |  |  |       tensor_Tensor.device_data(), | 
					
						
							|  |  |  |       static_cast<int>(tensor_Reduction.stride()[0]), | 
					
						
							|  |  |  |       static_cast<int>(tensor_Tensor.stride()[0]) | 
					
						
							|  |  |  |     ); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     // find workspace requirement for parallel split-k reduction
 | 
					
						
							|  |  |  |     size_t workspace_size = Conv2d::get_workspace_size(conv2d_args); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     cutlass::device_memory::allocation<uint8_t> workspace(workspace_size); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     cutlass::Status status = conv2d_op.initialize(conv2d_args, workspace.get()); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     if (status != cutlass::Status::kSuccess) { | 
					
						
							|  |  |  |       cudaError_t error = cudaGetLastError(); | 
					
						
							|  |  |  |       std::cerr << "This test is not supported: " << cudaGetErrorString(error) << "\n"; | 
					
						
							|  |  |  |       return true; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     // conv2d operation with parallel split-k-mode
 | 
					
						
							|  |  |  |     if (split_k_mode == cutlass::conv::SplitKMode::kParallel) { | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |       // conv2d output is written to workspace in global memory
 | 
					
						
							|  |  |  |       conv2d_args.ref_D.reset(reinterpret_cast<ElementC*>(workspace.get())); | 
					
						
							|  |  |  |       // accumulate mma for each cta in k-dimension (1.0 * A * B)
 | 
					
						
							|  |  |  |       conv2d_args.output_op = {ElementCompute(1), ElementCompute(0)};  | 
					
						
							|  |  |  |       // update conv2d operator arguments
 | 
					
						
							|  |  |  |       status = conv2d_op.update(conv2d_args, workspace.get()); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |      | 
					
						
							|  |  |  |     EXPECT_TRUE(status == cutlass::Status::kSuccess); | 
					
						
							|  |  |  |     if (status != cutlass::Status::kSuccess) { | 
					
						
							|  |  |  |       return false; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     // run conv2d operator
 | 
					
						
							|  |  |  |     status = conv2d_op(); | 
					
						
							|  |  |  |      | 
					
						
							|  |  |  |     EXPECT_TRUE(status == cutlass::Status::kSuccess); | 
					
						
							|  |  |  |     if (status != cutlass::Status::kSuccess) { | 
					
						
							|  |  |  |       return false; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     bool passed = false; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     cudaError_t result = cudaDeviceSynchronize(); | 
					
						
							|  |  |  |     EXPECT_EQ(result, cudaSuccess) << " device reference error: "  | 
					
						
							|  |  |  |                                    << cudaGetErrorString(result); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-05-07 01:24:51 +08:00
										 |  |  |     // Final reduction over the partial reduction tensor
 | 
					
						
							|  |  |  |     using Functor = cutlass::plus<ElementAccumulator>; | 
					
						
							|  |  |  |     using TensorReduction = cutlass::reduction::device::TensorReduction< | 
					
						
							|  |  |  |       ElementAccumulator, | 
					
						
							|  |  |  |       ElementAccumulator, | 
					
						
							|  |  |  |       LayoutC,  | 
					
						
							|  |  |  |       Functor, | 
					
						
							|  |  |  |       8, | 
					
						
							|  |  |  |       ElementAccumulator | 
					
						
							|  |  |  |     >; | 
					
						
							| 
									
										
										
										
											2021-09-21 02:02:22 +08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-05-07 01:24:51 +08:00
										 |  |  |     TensorReduction reduction(tensor_Reduction.extent(), 2); | 
					
						
							| 
									
										
										
										
											2021-09-21 02:02:22 +08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-05-07 01:24:51 +08:00
										 |  |  |     cutlass::DeviceAllocation<uint8_t> reduction_device_workspace(reduction.workspace_size()); | 
					
						
							| 
									
										
										
										
											2021-09-21 02:02:22 +08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-05-07 01:24:51 +08:00
										 |  |  |     status = reduction.reduce( | 
					
						
							|  |  |  |       tensor_Final_Reduction.device_ref(), | 
					
						
							|  |  |  |       tensor_Reduction.device_ref(), | 
					
						
							|  |  |  |       reduction_device_workspace.get(), | 
					
						
							|  |  |  |       ElementAccumulator()); | 
					
						
							| 
									
										
										
										
											2021-09-21 02:02:22 +08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-05-07 01:24:51 +08:00
										 |  |  |     EXPECT_EQ(status, cutlass::Status::kSuccess); | 
					
						
							|  |  |  |     EXPECT_EQ(cudaDeviceSynchronize(), cudaSuccess); | 
					
						
							| 
									
										
										
										
											2021-09-21 02:02:22 +08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-05-07 01:24:51 +08:00
										 |  |  |     //
 | 
					
						
							|  |  |  |     // Reference check
 | 
					
						
							|  |  |  |     //
 | 
					
						
							| 
									
										
										
										
											2021-09-21 02:02:22 +08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-05-07 01:24:51 +08:00
										 |  |  |     tensor_D_computed.sync_host(); | 
					
						
							| 
									
										
										
										
											2021-09-21 02:02:22 +08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-07-23 12:40:53 +08:00
										 |  |  | #if CUTLASS_CONV_TEST_UNIT_REFERENCE_DEVICE_ENABLED
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     cutlass::reference::device::Conv2d< | 
					
						
							|  |  |  |       ElementA, | 
					
						
							|  |  |  |       LayoutA, | 
					
						
							|  |  |  |       ElementB, | 
					
						
							|  |  |  |       LayoutB, | 
					
						
							|  |  |  |       ElementC, | 
					
						
							|  |  |  |       LayoutC, | 
					
						
							|  |  |  |       ElementCompute, | 
					
						
							|  |  |  |       ElementAccumulator  | 
					
						
							|  |  |  |     >( | 
					
						
							|  |  |  |       kConvolutionalOperator, | 
					
						
							|  |  |  |       problem_size, | 
					
						
							|  |  |  |       tensor_A.device_ref(), | 
					
						
							|  |  |  |       tensor_B.device_ref(), | 
					
						
							|  |  |  |       tensor_C.device_ref(), | 
					
						
							|  |  |  |       tensor_D_reference.device_ref(), | 
					
						
							|  |  |  |       alpha,  | 
					
						
							|  |  |  |       beta); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     // sync host (copy device data to host) for dumping error output in case of mismatches
 | 
					
						
							|  |  |  |     tensor_D_reference.sync_host(); | 
					
						
							|  |  |  |      | 
					
						
							|  |  |  | #else 
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     cutlass::reference::host::Conv2d< | 
					
						
							|  |  |  |       ElementA, | 
					
						
							|  |  |  |       LayoutA, | 
					
						
							|  |  |  |       ElementB, | 
					
						
							|  |  |  |       LayoutB, | 
					
						
							|  |  |  |       ElementC, | 
					
						
							|  |  |  |       LayoutC, | 
					
						
							|  |  |  |       ElementCompute, | 
					
						
							|  |  |  |       ElementAccumulator | 
					
						
							|  |  |  |     >( | 
					
						
							|  |  |  |       kConvolutionalOperator, | 
					
						
							|  |  |  |       problem_size, | 
					
						
							|  |  |  |       tensor_A.host_ref(), | 
					
						
							|  |  |  |       tensor_B.host_ref(), | 
					
						
							|  |  |  |       tensor_C.host_ref(), | 
					
						
							|  |  |  |       tensor_D_reference.host_ref(), | 
					
						
							|  |  |  |       alpha,  | 
					
						
							|  |  |  |       beta); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | #endif
 | 
					
						
							| 
									
										
										
										
											2021-09-21 02:02:22 +08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-05-07 01:24:51 +08:00
										 |  |  |     passed = cutlass::reference::host::TensorEquals( | 
					
						
							|  |  |  |       tensor_D_computed.host_view(),  | 
					
						
							|  |  |  |       tensor_D_reference.host_view()); | 
					
						
							| 
									
										
										
										
											2021-09-21 02:02:22 +08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-05-07 01:24:51 +08:00
										 |  |  |     EXPECT_TRUE(passed); | 
					
						
							| 
									
										
										
										
											2021-09-21 02:02:22 +08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-05-07 01:24:51 +08:00
										 |  |  |     //
 | 
					
						
							|  |  |  |     // Reference check on reduction results
 | 
					
						
							|  |  |  |     //
 | 
					
						
							| 
									
										
										
										
											2021-09-21 02:02:22 +08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-05-07 01:24:51 +08:00
										 |  |  |     tensor_Reduction.sync_host(); | 
					
						
							|  |  |  |     tensor_Final_Reduction.sync_host(); | 
					
						
							| 
									
										
										
										
											2021-09-21 02:02:22 +08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-05-07 01:24:51 +08:00
										 |  |  |     // compute backwards for reduction results
 | 
					
						
							|  |  |  |     cutlass::HostTensor<ElementAccumulator, LayoutC> reference_Reduction; | 
					
						
							|  |  |  |     reference_Reduction.resize({ | 
					
						
							|  |  |  |       1, | 
					
						
							|  |  |  |       1, | 
					
						
							|  |  |  |       1, | 
					
						
							|  |  |  |       (problem_size.K)  | 
					
						
							|  |  |  |     }); | 
					
						
							| 
									
										
										
										
											2021-09-21 02:02:22 +08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-05-07 01:24:51 +08:00
										 |  |  |     for (int k = 0; k < problem_size.K; ++k) { | 
					
						
							|  |  |  |       ElementAccumulator reduced_value = ElementAccumulator(); | 
					
						
							|  |  |  |       for (int n = 0; n < problem_size.N; ++n) { | 
					
						
							|  |  |  |         for (int p = 0; p < problem_size.P; ++p) { | 
					
						
							|  |  |  |           for (int q = 0; q < problem_size.Q; ++q) { | 
					
						
							|  |  |  |             reduced_value += tensor_D_reference.at({n, p, q, k}); | 
					
						
							|  |  |  |           } | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |       } | 
					
						
							|  |  |  |       reference_Reduction.at({0, 0, 0, k}) = reduced_value; | 
					
						
							| 
									
										
										
										
											2021-09-21 02:02:22 +08:00
										 |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-05-07 01:24:51 +08:00
										 |  |  |     passed = cutlass::reference::host::TensorEquals( | 
					
						
							|  |  |  |       tensor_Final_Reduction.host_view(), | 
					
						
							|  |  |  |       reference_Reduction.host_view() | 
					
						
							|  |  |  |     ); | 
					
						
							| 
									
										
										
										
											2021-07-23 12:40:53 +08:00
										 |  |  | 
 | 
					
						
							|  |  |  |     EXPECT_TRUE(passed); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     if (!passed) { | 
					
						
							|  |  |  |       std::stringstream fname; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |       fname << "error_Conv2d_ImplicitGemm_device_" | 
					
						
							|  |  |  |         << (split_k_mode == cutlass::conv::SplitKMode::kSerial ? "serial_reduction_" : "parallel_reduction_") | 
					
						
							|  |  |  |         << (Conv2d::kConvolutionalOperator == cutlass::conv::Operator::kFprop ? "fprop_" : | 
					
						
							|  |  |  |             (Conv2d::kConvolutionalOperator == cutlass::conv::Operator::kDgrad ? "dgrad_" : "wgrad_"))  | 
					
						
							|  |  |  |         << "nhwc_" | 
					
						
							|  |  |  |         << problem_size.N << "x" | 
					
						
							|  |  |  |         << problem_size.H << "x" | 
					
						
							|  |  |  |         << problem_size.W << "x" | 
					
						
							|  |  |  |         << problem_size.C  | 
					
						
							|  |  |  |         << "_krsc_" | 
					
						
							|  |  |  |         << problem_size.K << "x" | 
					
						
							|  |  |  |         << problem_size.R << "x" | 
					
						
							|  |  |  |         << problem_size.S << "x" | 
					
						
							|  |  |  |         << problem_size.C  | 
					
						
							|  |  |  |         << "_padding_"  | 
					
						
							|  |  |  |         << problem_size.pad_h << "x" | 
					
						
							|  |  |  |         << problem_size.pad_w  | 
					
						
							|  |  |  |         << "_stride_"   | 
					
						
							|  |  |  |         << problem_size.stride_h << "x" | 
					
						
							|  |  |  |         << problem_size.stride_w  | 
					
						
							|  |  |  |         << "_dilation_" | 
					
						
							|  |  |  |         << problem_size.dilation_h << "x" | 
					
						
							|  |  |  |         << problem_size.dilation_w << "_" | 
					
						
							|  |  |  |         << (problem_size.mode == cutlass::conv::Mode::kCrossCorrelation ? "xcorr_" : "conv_") | 
					
						
							|  |  |  |         << Conv2d::ThreadblockShape::kM << "x"   | 
					
						
							|  |  |  |         << Conv2d::ThreadblockShape::kN << "x"   | 
					
						
							|  |  |  |         << Conv2d::ThreadblockShape::kK << "_" | 
					
						
							|  |  |  |         << Conv2d::WarpShape::kM << "x"   | 
					
						
							|  |  |  |         << Conv2d::WarpShape::kN << "x"   | 
					
						
							|  |  |  |         << Conv2d::WarpShape::kK << ".txt"; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |       std::cout << fname.str() << std::endl; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |       std::ofstream results(fname.str()); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |       results << problem_size << std::endl; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |       results | 
					
						
							|  |  |  |         << "\nA:\n" << tensor_A.host_view() << "\n" | 
					
						
							|  |  |  |         << "\nB:\n" << tensor_B.host_view() << "\n" | 
					
						
							|  |  |  |         << "\nC:\n" << tensor_C.host_view() << "\n" | 
					
						
							|  |  |  |         << "\nD reference:\n" << tensor_D_reference.host_view() << "\n" | 
					
						
							| 
									
										
										
										
											2022-05-07 01:24:51 +08:00
										 |  |  |         << "\nD computed:\n" << tensor_D_computed.host_view() << "\n" | 
					
						
							|  |  |  |         << "\nreduction reference:\n" << reference_Reduction.host_view() << "\n" | 
					
						
							|  |  |  |         << "\nreduction computed:\n" << tensor_Reduction.host_view() << "\n"; | 
					
						
							| 
									
										
										
										
											2021-07-23 12:40:53 +08:00
										 |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     return passed; | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  | }; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | /////////////////////////////////////////////////////////////////////////////////////////////////////////
 | 
					
						
							|  |  |  | // TestAllConv: Runs cutlass::conv::device::ImplicitGemmConvolution operator and compares it with reference
 | 
					
						
							|  |  |  | // TestAllConv runs conv operator on default conv problem sizes from test::conv::device::TestbedConv2dProblemSizes
 | 
					
						
							|  |  |  | // Additionaly, each conv2d test can provide conv problem sizes (conv_test_sizes) and blacklist of sizes 
 | 
					
						
							|  |  |  | // (conv_blacklist_sizes)
 | 
					
						
							|  |  |  | /////////////////////////////////////////////////////////////////////////////////////////////////////////////
 | 
					
						
							|  |  |  | template <typename ImplicitGemm> | 
					
						
							|  |  |  | bool TestAllConv2dWithReduction( | 
					
						
							|  |  |  |   const Conv2dProblemVector & conv_test_sizes = Conv2dProblemVector(), | 
					
						
							|  |  |  |   const Conv2dProblemVector & conv_blacklist_sizes = Conv2dProblemVector()) { | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   bool passed = true; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   //
 | 
					
						
							|  |  |  |   // Testbed object
 | 
					
						
							|  |  |  |   //
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   TestbedConv2dWithReduction<ImplicitGemm> testbed; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   //
 | 
					
						
							|  |  |  |   // Get conv problem sizes to run conv operator 
 | 
					
						
							|  |  |  |   //
 | 
					
						
							|  |  |  |   TestbedConv2dProblemSizes conv_problems(128/cutlass::sizeof_bits<typename ImplicitGemm::ElementA>::value); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   // Vector of conv2d problem sizes to avoid duplicate runs
 | 
					
						
							|  |  |  |   Conv2dProblemVector conv_tested_sizes; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   Conv2dProblemVector const *problem_vectors[] = { | 
					
						
							|  |  |  |     &conv_test_sizes,                               // run user specified sizes
 | 
					
						
							|  |  |  |     &conv_problems.conv2d_default_sizes,            // run default and cudnn bug sizes
 | 
					
						
							|  |  |  |     &conv_problems.conv2d_resnet50_sizes,           // run resnet50 sizes
 | 
					
						
							|  |  |  | #if CUTLASS_CONV_UNIT_TEST_RIGOROUS_SIZE_ENABLED 
 | 
					
						
							|  |  |  |     &conv_problems.conv2d_rigorous_sizes,           // run large and rigorous sizes if enabled
 | 
					
						
							|  |  |  | #endif
 | 
					
						
							|  |  |  |   }; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   // Sweep conv2d problem sizes (split-k-mode=kSerial, split-k-slice=1, alpha=1.0, beta=0.0)
 | 
					
						
							|  |  |  |   for (Conv2dProblemVector const * problem_vector : problem_vectors) { | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     //  Run conv testbed on default convolution sizes
 | 
					
						
							|  |  |  |     for(auto conv_problem : *problem_vector) { | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |       // Skip blacklist and avoid duplicate problem sizes
 | 
					
						
							|  |  |  |       if (std::find(conv_blacklist_sizes.begin(), conv_blacklist_sizes.end(), conv_problem) != conv_blacklist_sizes.end() || | 
					
						
							|  |  |  |           std::find(conv_tested_sizes.begin(), conv_tested_sizes.end(), conv_problem) != conv_tested_sizes.end()) { | 
					
						
							|  |  |  |         continue; | 
					
						
							|  |  |  |       } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |       //
 | 
					
						
							|  |  |  |       // Procedurally disable certain cases
 | 
					
						
							|  |  |  |       //
 | 
					
						
							|  |  |  |    | 
					
						
							|  |  |  |       // CUTLASS DGRAD's *unity* stride specialization only support stride {1, 1} 
 | 
					
						
							|  |  |  |       if ((ImplicitGemm::kConvolutionalOperator ==  | 
					
						
							|  |  |  |             cutlass::conv::Operator::kDgrad) &&  | 
					
						
							|  |  |  |           (ImplicitGemm::ImplicitGemmKernel::Mma::IteratorA::kStrideSupport ==  | 
					
						
							|  |  |  |             cutlass::conv::StrideSupport::kUnity)) { | 
					
						
							|  |  |  |         if (!((conv_problem.stride_h == 1) && (conv_problem.stride_w == 1))) { | 
					
						
							|  |  |  |           continue; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |       } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | #if 0 // relax restrictions on analytic strided dgrad
 | 
					
						
							|  |  |  |       // CUTLASS DGRAD's *strided* specialization only support stride >= {2, 2} 
 | 
					
						
							|  |  |  |       if ((ImplicitGemm::kConvolutionalOperator ==  | 
					
						
							|  |  |  |             cutlass::conv::Operator::kDgrad) &&  | 
					
						
							|  |  |  |           (ImplicitGemm::ImplicitGemmKernel::Mma::IteratorA::kStrideSupport ==  | 
					
						
							|  |  |  |             cutlass::conv::StrideSupport::kStrided)) { | 
					
						
							|  |  |  |          if (((conv_problem.stride_h == 1) && (conv_problem.stride_w == 1))) { | 
					
						
							|  |  |  |            continue; | 
					
						
							|  |  |  |          } | 
					
						
							|  |  |  |       } | 
					
						
							|  |  |  | #endif
 | 
					
						
							|  |  |  |        | 
					
						
							|  |  |  |       //
 | 
					
						
							|  |  |  |       // Test
 | 
					
						
							|  |  |  |       //
 | 
					
						
							|  |  |  |       // push back tested problem size to avoid re-running duplicates
 | 
					
						
							|  |  |  |       conv_tested_sizes.push_back(conv_problem); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |       // test mode = xcross
 | 
					
						
							|  |  |  |       passed = testbed.run( | 
					
						
							|  |  |  |         conv_problem, | 
					
						
							|  |  |  |         cutlass::conv::SplitKMode::kSerial); | 
					
						
							|  |  |  |      | 
					
						
							|  |  |  |       if (!passed) { | 
					
						
							|  |  |  |         return false; | 
					
						
							|  |  |  |       } | 
					
						
							|  |  |  |        | 
					
						
							|  |  |  |       // test mode = convolution
 | 
					
						
							|  |  |  |       passed = testbed.run( | 
					
						
							|  |  |  |         conv_problem.reset_mode(cutlass::conv::Mode::kConvolution), | 
					
						
							|  |  |  |         cutlass::conv::SplitKMode::kSerial); | 
					
						
							|  |  |  |      | 
					
						
							|  |  |  |       if (!passed) { | 
					
						
							|  |  |  |         return false; | 
					
						
							|  |  |  |       } | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   // CUTLASS DGRAD's *strided* specialization does not support split-k mode 
 | 
					
						
							|  |  |  |   if ((ImplicitGemm::kConvolutionalOperator ==  | 
					
						
							|  |  |  |           cutlass::conv::Operator::kDgrad) &&  | 
					
						
							|  |  |  |       (ImplicitGemm::ImplicitGemmKernel::Mma::IteratorA::kStrideSupport ==  | 
					
						
							|  |  |  |         cutlass::conv::StrideSupport::kStrided)) { | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     passed = testbed.run( | 
					
						
							|  |  |  |       cutlass::conv::Conv2dProblemSize( | 
					
						
							|  |  |  |       {1, 56, 56, 8},   // input size (NHWC)
 | 
					
						
							|  |  |  |       {8, 1, 1, 8},     // filter size (KRSC)
 | 
					
						
							|  |  |  |       {0, 0, 0, 0},     // padding (pad_h, _, pad_w, _)
 | 
					
						
							|  |  |  |       {2, 2},           // stride (stride_h, stride_w)
 | 
					
						
							|  |  |  |       {1, 1}),          // dilation (dilation_h, dilation_w)
 | 
					
						
							|  |  |  |       cutlass::conv::SplitKMode::kSerial, | 
					
						
							|  |  |  |       cutlass::from_real<typename ImplicitGemm::ElementCompute>(2.0),  | 
					
						
							|  |  |  |       cutlass::from_real<typename ImplicitGemm::ElementCompute>(2.0)); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     if (!passed) { | 
					
						
							|  |  |  |       return false; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     return passed; | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   // Sweep split-k-slice using serial and prallel reduction with non-unity alpha and non-zero beta for 
 | 
					
						
							|  |  |  |   // a single conv2d problem size. Convolution unit tests take a long time to run so only sweep parameters 
 | 
					
						
							|  |  |  |   // which are abolutely neccessary to catch functional bugs. The below code does provide option to sweep 
 | 
					
						
							|  |  |  |   // alpha and beta for local testing, but only runs one value for alpha and beta.
 | 
					
						
							|  |  |  |   cutlass::conv::Conv2dProblemSize conv2d_split_k_test_size ( | 
					
						
							|  |  |  |       {1, 17, 11, 288},   // input size (NHWC)
 | 
					
						
							|  |  |  |       {160, 3, 3, 288},   // filter size (KRSC)
 | 
					
						
							|  |  |  |       {1, 1, 1, 1},       // padding (pad_h, _, pad_w, _)
 | 
					
						
							|  |  |  |       {1, 1},             // stride (stride_h, stride_w)
 | 
					
						
							|  |  |  |       {1, 1}              // dilation (dilation_h, dilation_w)
 | 
					
						
							|  |  |  |     ); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-05-07 01:24:51 +08:00
										 |  |  |   // Parallel SplitK is not tested.
 | 
					
						
							| 
									
										
										
										
											2021-07-23 12:40:53 +08:00
										 |  |  |   cutlass::conv::SplitKMode split_k_modes [] = { | 
					
						
							|  |  |  |     cutlass::conv::SplitKMode::kSerial, | 
					
						
							|  |  |  |   }; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   int split_k_slices[] = { | 
					
						
							|  |  |  |     1, 2, 3, 4, 201 | 
					
						
							|  |  |  |   }; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   double problem_alpha[] = { | 
					
						
							|  |  |  |     2.0 | 
					
						
							|  |  |  |   }; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   double problem_beta[] = { | 
					
						
							|  |  |  |     2.0 | 
					
						
							|  |  |  |   }; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   for (auto split_k_mode : split_k_modes) { | 
					
						
							|  |  |  |     for (auto split_k_slice : split_k_slices) { | 
					
						
							|  |  |  |       for (auto alpha : problem_alpha) { | 
					
						
							|  |  |  |         for (auto beta : problem_beta) { | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |           passed = testbed.run( | 
					
						
							|  |  |  |             conv2d_split_k_test_size.reset_split_k_slices(split_k_slice), | 
					
						
							|  |  |  |             split_k_mode, | 
					
						
							|  |  |  |             cutlass::from_real<typename ImplicitGemm::ElementCompute>(alpha),  | 
					
						
							|  |  |  |             cutlass::from_real<typename ImplicitGemm::ElementCompute>(beta)); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |           if (!passed) { | 
					
						
							|  |  |  |             return false; | 
					
						
							|  |  |  |           } | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |       } | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   return passed; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | /////////////////////////////////////////////////////////////////////////////////////////////////
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | } // namespace device
 | 
					
						
							|  |  |  | } // namespace conv
 | 
					
						
							|  |  |  | } // namespace test
 |