mirror of
https://github.com/ROCm/composable_kernel.git
synced 2026-05-03 13:11:25 +00:00
Layernorm and groupnorm support to save mean and inverse std in forward (#929)
* save mean and inverse std in normalization * Save mean and inverse std in splitK * Vector save mean and inv std * Modify instance for save mean and std * simplify the layernorm example * Save mean and std in groupnorm example * Save mean and inv std in ckProfiler and test * Remove compute data type from base class * Save mean and inv std in client example * Add changelog * clang format * Fix compile error * Refine naming * Avoid error in bf16 * revert changelog
This commit is contained in:
@@ -28,6 +28,7 @@ template <typename XDataType,
|
||||
typename BetaDataType,
|
||||
typename ComputeDataType,
|
||||
typename YDataType,
|
||||
typename SaveMeanInvStdDataType,
|
||||
typename YElementwiseOperation,
|
||||
index_t Rank,
|
||||
index_t NumReduceDim,
|
||||
@@ -43,12 +44,13 @@ template <typename XDataType,
|
||||
index_t BetaSrcVectorDim,
|
||||
index_t BetaSrcVectorSize,
|
||||
index_t YDstVectorSize,
|
||||
index_t SaveMeanInvStdDstVectorSize,
|
||||
bool UseWelford = true>
|
||||
struct DeviceNormalizationImpl : public DeviceNormalization<XDataType,
|
||||
GammaDataType,
|
||||
BetaDataType,
|
||||
ComputeDataType,
|
||||
YDataType,
|
||||
SaveMeanInvStdDataType,
|
||||
YElementwiseOperation,
|
||||
Rank,
|
||||
NumReduceDim>
|
||||
@@ -64,18 +66,24 @@ struct DeviceNormalizationImpl : public DeviceNormalization<XDataType,
|
||||
(BetaSrcVectorDim == 1 && KThreadSliceSize % BetaSrcVectorSize == 0)),
|
||||
"Invalid thread slice sizes and/or beta vector sizes configuration, please check!");
|
||||
|
||||
static_assert(MThreadSliceSize % SaveMeanInvStdDstVectorSize == 0,
|
||||
"Invalid thread slice sizes and/or save mean and inverse std vector sizes "
|
||||
"configuration, please check!");
|
||||
|
||||
using PassThrough = tensor_operation::element_wise::PassThrough;
|
||||
|
||||
static constexpr index_t NumInvariantDim = Rank - NumReduceDim;
|
||||
static constexpr index_t M_BlockTileSize = MThreadClusterSize * MThreadSliceSize;
|
||||
static constexpr index_t K_BlockTileSize = KThreadClusterSize * KThreadSliceSize;
|
||||
|
||||
static constexpr bool reduceAllDim = (NumInvariantDim == 0);
|
||||
static_assert(!reduceAllDim); // TODO
|
||||
|
||||
static auto MakeSrc2dDescriptor(const std::vector<index_t>& inLengths,
|
||||
const std::vector<index_t>& inStrides,
|
||||
int numBlockTileIteration)
|
||||
{
|
||||
constexpr index_t NumInvariantDim = Rank - NumReduceDim;
|
||||
static constexpr index_t numSrcDim = Rank;
|
||||
static constexpr bool reduceAllDim = (NumInvariantDim == 0);
|
||||
|
||||
const auto tupleSrcLengths = make_tuple_from_array(inLengths, Number<numSrcDim>{});
|
||||
const auto tupleSrcStrides = make_tuple_from_array(inStrides, Number<numSrcDim>{});
|
||||
@@ -133,7 +141,37 @@ struct DeviceNormalizationImpl : public DeviceNormalization<XDataType,
|
||||
return (in_grid_desc_m_k_padded);
|
||||
};
|
||||
|
||||
static auto MakeSaveMeanInvStdDescriptor_M(const std::vector<index_t>& lengths,
|
||||
const std::vector<index_t>& strides)
|
||||
{
|
||||
using InvariantDims = typename arithmetic_sequence_gen<0, NumInvariantDim, 1>::type;
|
||||
|
||||
const auto tupleSrcLengths = make_tuple_from_array_and_index_seq(lengths, InvariantDims{});
|
||||
const auto tupleSrcStrides = make_tuple_from_array_and_index_seq(strides, InvariantDims{});
|
||||
|
||||
const auto desc = make_naive_tensor_descriptor(tupleSrcLengths, tupleSrcStrides);
|
||||
|
||||
const auto grid_desc_m =
|
||||
transform_tensor_descriptor(desc,
|
||||
make_tuple(make_merge_transform(tupleSrcLengths)),
|
||||
make_tuple(InvariantDims{}),
|
||||
make_tuple(Sequence<0>{}));
|
||||
|
||||
const auto invariantLength = grid_desc_m.GetLength(Number<0>{});
|
||||
const auto pad_M =
|
||||
math::integer_least_multiple(invariantLength, M_BlockTileSize) - invariantLength;
|
||||
|
||||
auto grid_desc_m_padded = transform_tensor_descriptor(
|
||||
grid_desc_m,
|
||||
make_tuple(make_right_pad_transform(invariantLength, pad_M)),
|
||||
make_tuple(Sequence<0>{}),
|
||||
make_tuple(Sequence<0>{}));
|
||||
|
||||
return grid_desc_m_padded;
|
||||
}
|
||||
|
||||
using GridDesc_M_K = decltype(MakeSrc2dDescriptor({1}, {1}, 1));
|
||||
using GridDesc_M = decltype(MakeSaveMeanInvStdDescriptor_M({1}, {1}));
|
||||
|
||||
struct Argument : public BaseArgument
|
||||
{
|
||||
@@ -142,17 +180,23 @@ struct DeviceNormalizationImpl : public DeviceNormalization<XDataType,
|
||||
const std::vector<index_t> gammaStrides,
|
||||
const std::vector<index_t> betaStrides,
|
||||
const std::vector<index_t> yStrides,
|
||||
const std::vector<index_t> saveMeanStrides,
|
||||
const std::vector<index_t> saveInvStdStrides,
|
||||
const std::vector<index_t> reduceDims,
|
||||
YElementwiseOperation y_elementwise_op,
|
||||
double epsilon,
|
||||
const XDataType* p_x,
|
||||
const GammaDataType* p_gamma,
|
||||
const BetaDataType* p_beta,
|
||||
YDataType* p_y)
|
||||
YDataType* p_y,
|
||||
SaveMeanInvStdDataType* p_saveMean,
|
||||
SaveMeanInvStdDataType* p_saveInvStd)
|
||||
: p_x_(p_x),
|
||||
p_gamma_(p_gamma),
|
||||
p_beta_(p_beta),
|
||||
p_y_(p_y),
|
||||
p_saveMean_(p_saveMean),
|
||||
p_saveInvStd_(p_saveInvStd),
|
||||
y_elementwise_op_(y_elementwise_op)
|
||||
{
|
||||
epsilon_ = static_cast<ComputeDataType>(epsilon);
|
||||
@@ -162,16 +206,14 @@ struct DeviceNormalizationImpl : public DeviceNormalization<XDataType,
|
||||
yStrides_ = shuffle_tensor_dimensions<Rank, NumReduceDim>(yStrides, reduceDims);
|
||||
gammaStrides_ = shuffle_tensor_dimensions<Rank, NumReduceDim>(gammaStrides, reduceDims);
|
||||
betaStrides_ = shuffle_tensor_dimensions<Rank, NumReduceDim>(betaStrides, reduceDims);
|
||||
saveMeanStrides_ = saveMeanStrides;
|
||||
saveInvStdStrides_ = saveInvStdStrides;
|
||||
|
||||
long_index_t invariant_length;
|
||||
long_index_t reduce_length;
|
||||
std::tie(MRaw_, KRaw_) = get_2d_lengths<Rank, NumReduceDim>(Lengths_);
|
||||
|
||||
std::tie(invariant_length, reduce_length) =
|
||||
get_2d_lengths<Rank, NumReduceDim>(Lengths_);
|
||||
numBlockTileIteration_ = math::integer_divide_ceil(KRaw_, K_BlockTileSize);
|
||||
|
||||
numBlockTileIteration_ = math::integer_divide_ceil(reduce_length, K_BlockTileSize);
|
||||
|
||||
gridSize_ = math::integer_divide_ceil(invariant_length, M_BlockTileSize);
|
||||
gridSize_ = math::integer_divide_ceil(MRaw_, M_BlockTileSize);
|
||||
|
||||
x_grid_desc_m_k_ = MakeSrc2dDescriptor(Lengths_, xStrides_, numBlockTileIteration_);
|
||||
gamma_grid_desc_m_k_ =
|
||||
@@ -179,9 +221,16 @@ struct DeviceNormalizationImpl : public DeviceNormalization<XDataType,
|
||||
beta_grid_desc_m_k_ =
|
||||
MakeSrc2dDescriptor(Lengths_, betaStrides_, numBlockTileIteration_);
|
||||
y_grid_desc_m_k_ = MakeSrc2dDescriptor(Lengths_, yStrides_, numBlockTileIteration_);
|
||||
save_mean_grid_desc_m_ = MakeSaveMeanInvStdDescriptor_M(Lengths_, saveMeanStrides);
|
||||
save_inv_std_grid_desc_m_ = MakeSaveMeanInvStdDescriptor_M(Lengths_, saveInvStdStrides);
|
||||
|
||||
isSweeponce_ =
|
||||
x_grid_desc_m_k_.GetLength(Number<1>{}) <= KThreadClusterSize * KThreadSliceSize;
|
||||
|
||||
if constexpr(NumInvariantDim == 0)
|
||||
invariant_lowest_length_ = 1;
|
||||
else
|
||||
invariant_lowest_length_ = Lengths_[NumInvariantDim - 1];
|
||||
}
|
||||
|
||||
ComputeDataType epsilon_;
|
||||
@@ -190,12 +239,16 @@ struct DeviceNormalizationImpl : public DeviceNormalization<XDataType,
|
||||
const GammaDataType* p_gamma_;
|
||||
const BetaDataType* p_beta_;
|
||||
YDataType* p_y_;
|
||||
SaveMeanInvStdDataType* p_saveMean_;
|
||||
SaveMeanInvStdDataType* p_saveInvStd_;
|
||||
|
||||
std::vector<index_t> Lengths_;
|
||||
std::vector<index_t> xStrides_;
|
||||
std::vector<index_t> gammaStrides_;
|
||||
std::vector<index_t> betaStrides_;
|
||||
std::vector<index_t> yStrides_;
|
||||
std::vector<index_t> saveMeanStrides_;
|
||||
std::vector<index_t> saveInvStdStrides_;
|
||||
|
||||
YElementwiseOperation y_elementwise_op_;
|
||||
|
||||
@@ -206,7 +259,14 @@ struct DeviceNormalizationImpl : public DeviceNormalization<XDataType,
|
||||
GridDesc_M_K gamma_grid_desc_m_k_;
|
||||
GridDesc_M_K beta_grid_desc_m_k_;
|
||||
GridDesc_M_K y_grid_desc_m_k_;
|
||||
GridDesc_M save_mean_grid_desc_m_;
|
||||
GridDesc_M save_inv_std_grid_desc_m_;
|
||||
bool isSweeponce_;
|
||||
|
||||
index_t MRaw_; // invarient length
|
||||
index_t KRaw_; // reduce length
|
||||
|
||||
index_t invariant_lowest_length_;
|
||||
};
|
||||
|
||||
struct Invoker : public BaseInvoker
|
||||
@@ -217,9 +277,11 @@ struct DeviceNormalizationImpl : public DeviceNormalization<XDataType,
|
||||
GammaDataType,
|
||||
BetaDataType,
|
||||
YDataType,
|
||||
SaveMeanInvStdDataType,
|
||||
ComputeDataType,
|
||||
YElementwiseOperation,
|
||||
GridDesc_M_K,
|
||||
GridDesc_M,
|
||||
BlockSize,
|
||||
MThreadClusterSize,
|
||||
KThreadClusterSize,
|
||||
@@ -233,6 +295,7 @@ struct DeviceNormalizationImpl : public DeviceNormalization<XDataType,
|
||||
BetaSrcVectorSize,
|
||||
XYSrcVectorDim,
|
||||
YDstVectorSize,
|
||||
SaveMeanInvStdDstVectorSize,
|
||||
UseWelford>(arg.isSweeponce_);
|
||||
|
||||
float avg_time = 0;
|
||||
@@ -245,12 +308,16 @@ struct DeviceNormalizationImpl : public DeviceNormalization<XDataType,
|
||||
arg.gamma_grid_desc_m_k_,
|
||||
arg.beta_grid_desc_m_k_,
|
||||
arg.y_grid_desc_m_k_,
|
||||
arg.save_mean_grid_desc_m_,
|
||||
arg.save_inv_std_grid_desc_m_,
|
||||
arg.numBlockTileIteration_,
|
||||
arg.epsilon_,
|
||||
arg.p_x_,
|
||||
arg.p_gamma_,
|
||||
arg.p_beta_,
|
||||
arg.p_y_,
|
||||
arg.p_saveMean_,
|
||||
arg.p_saveInvStd_,
|
||||
arg.y_elementwise_op_);
|
||||
|
||||
return (avg_time);
|
||||
@@ -267,8 +334,6 @@ struct DeviceNormalizationImpl : public DeviceNormalization<XDataType,
|
||||
{
|
||||
const Argument* p_arg_ = dynamic_cast<const Argument*>(p_arg);
|
||||
|
||||
constexpr index_t NumInvariantDim = Rank - NumReduceDim;
|
||||
|
||||
if constexpr(XYSrcVectorDim == 0)
|
||||
{
|
||||
if constexpr(NumInvariantDim == 0)
|
||||
@@ -277,13 +342,15 @@ struct DeviceNormalizationImpl : public DeviceNormalization<XDataType,
|
||||
}
|
||||
else
|
||||
{
|
||||
printf("!!!! %d\n", p_arg_->invariant_lowest_length_);
|
||||
|
||||
if(p_arg_->xStrides_[NumInvariantDim - 1] != 1)
|
||||
return false;
|
||||
|
||||
if(p_arg_->invariant_lowest_length % XSrcVectorSize != 0)
|
||||
if(p_arg_->invariant_lowest_length_ % XSrcVectorSize != 0)
|
||||
return false;
|
||||
|
||||
if(p_arg_->invariant_lowest_length % YDstVectorSize != 0)
|
||||
if(p_arg_->invariant_lowest_length_ % YDstVectorSize != 0)
|
||||
return false;
|
||||
};
|
||||
}
|
||||
@@ -325,7 +392,7 @@ struct DeviceNormalizationImpl : public DeviceNormalization<XDataType,
|
||||
if(p_arg_->betaStrides_[NumInvariantDim - 1] != 1)
|
||||
return (false);
|
||||
|
||||
if(p_arg_->invariant_lowest_length % BetaSrcVectorSize != 0)
|
||||
if(p_arg_->invariant_lowest_length_ % BetaSrcVectorSize != 0)
|
||||
return (false);
|
||||
}
|
||||
else // if fastest dim is reduced
|
||||
@@ -337,6 +404,9 @@ struct DeviceNormalizationImpl : public DeviceNormalization<XDataType,
|
||||
return (false);
|
||||
}
|
||||
|
||||
if(p_arg_->invariant_lowest_length_ % SaveMeanInvStdDstVectorSize != 0)
|
||||
return false;
|
||||
|
||||
return true;
|
||||
};
|
||||
|
||||
@@ -346,6 +416,8 @@ struct DeviceNormalizationImpl : public DeviceNormalization<XDataType,
|
||||
const std::vector<index_t> gammaStrides,
|
||||
const std::vector<index_t> betaStrides,
|
||||
const std::vector<index_t> yStrides,
|
||||
const std::vector<index_t> saveMeanStrides,
|
||||
const std::vector<index_t> saveInvStdStrides,
|
||||
const std::vector<index_t> reduceDims,
|
||||
double epsilon,
|
||||
const void* p_x,
|
||||
@@ -353,27 +425,30 @@ struct DeviceNormalizationImpl : public DeviceNormalization<XDataType,
|
||||
const void* p_beta,
|
||||
void* p_y,
|
||||
void* p_saveMean,
|
||||
void* p_saveInvVar,
|
||||
void* p_saveInvStd,
|
||||
YElementwiseOperation y_elementwise_op) override
|
||||
{
|
||||
// TODO
|
||||
// Optional cache of the intermediate results (mean and InvVariance) during the
|
||||
// forward pass could speedup in the backward
|
||||
ignore = p_saveMean;
|
||||
ignore = p_saveInvVar;
|
||||
if(lengths.size() != Rank || xStrides.size() != Rank || gammaStrides.size() != Rank ||
|
||||
betaStrides.size() != Rank || yStrides.size() != Rank ||
|
||||
saveMeanStrides.size() != NumInvariantDim || saveInvStdStrides.size() != NumInvariantDim)
|
||||
throw std::runtime_error("dimension is incorrect");
|
||||
|
||||
return std::make_unique<Argument>(lengths,
|
||||
xStrides,
|
||||
gammaStrides,
|
||||
betaStrides,
|
||||
yStrides,
|
||||
saveMeanStrides,
|
||||
saveInvStdStrides,
|
||||
reduceDims,
|
||||
y_elementwise_op,
|
||||
epsilon,
|
||||
static_cast<const XDataType*>(p_x),
|
||||
static_cast<const GammaDataType*>(p_gamma),
|
||||
static_cast<const BetaDataType*>(p_beta),
|
||||
static_cast<YDataType*>(p_y));
|
||||
static_cast<YDataType*>(p_y),
|
||||
static_cast<SaveMeanInvStdDataType*>(p_saveMean),
|
||||
static_cast<SaveMeanInvStdDataType*>(p_saveInvStd));
|
||||
};
|
||||
|
||||
std::unique_ptr<BaseInvoker> MakeInvokerPointer() override
|
||||
|
||||
@@ -19,7 +19,7 @@
|
||||
namespace ck {
|
||||
template <typename GridwiseWelford,
|
||||
typename XDataType,
|
||||
typename MeanVarDataType,
|
||||
typename WorkspaceMeanVarDataType,
|
||||
typename ComputeDataType,
|
||||
typename XGridDesc_M_K,
|
||||
typename MeanVarGridDesc_M_KBlock>
|
||||
@@ -28,8 +28,8 @@ kernel_normalizationSplitK1st(const XGridDesc_M_K x_grid_desc_m_k,
|
||||
const MeanVarGridDesc_M_KBlock mean_var_grid_desc_m_kblock,
|
||||
index_t num_k_block_tile_iteration,
|
||||
const XDataType* const __restrict__ p_x_global,
|
||||
MeanVarDataType* const __restrict__ p_welford_mean,
|
||||
MeanVarDataType* const __restrict__ p_welford_variance,
|
||||
WorkspaceMeanVarDataType* const __restrict__ p_welford_mean,
|
||||
WorkspaceMeanVarDataType* const __restrict__ p_welford_variance,
|
||||
int32_t* const __restrict__ p_welford_count)
|
||||
{
|
||||
GridwiseWelford::Run(x_grid_desc_m_k,
|
||||
@@ -42,16 +42,18 @@ kernel_normalizationSplitK1st(const XGridDesc_M_K x_grid_desc_m_k,
|
||||
};
|
||||
|
||||
template <typename GridwiseWelfordNormalization,
|
||||
typename MeanVarDataType,
|
||||
typename WorkspaceMeanVarDataType,
|
||||
typename XDataType,
|
||||
typename GammaDataType,
|
||||
typename BetaDataType,
|
||||
typename YDataType,
|
||||
typename SaveMeanInvStdDataType,
|
||||
typename ComputeDataType,
|
||||
typename YElementwiseOperation,
|
||||
typename MeanVarGridDesc_M_KBlock,
|
||||
typename CountGridDesc_M_KBlock,
|
||||
typename XYGammaBetaGridDesc_M_K>
|
||||
typename XYGammaBetaGridDesc_M_K,
|
||||
typename SaveMeanInvStdGridDesc_M>
|
||||
__global__ void
|
||||
kernel_normalizationSplitK2nd(const MeanVarGridDesc_M_KBlock mean_var_grid_desc_m_kblock,
|
||||
const CountGridDesc_M_KBlock count_grid_desc_m_kblock,
|
||||
@@ -59,17 +61,21 @@ kernel_normalizationSplitK2nd(const MeanVarGridDesc_M_KBlock mean_var_grid_desc_
|
||||
const XYGammaBetaGridDesc_M_K gamma_grid_desc_m_k,
|
||||
const XYGammaBetaGridDesc_M_K beta_grid_desc_m_k,
|
||||
const XYGammaBetaGridDesc_M_K y_grid_desc_m_k,
|
||||
const SaveMeanInvStdGridDesc_M save_mean_grid_desc_m,
|
||||
const SaveMeanInvStdGridDesc_M save_inv_std_grid_desc_m,
|
||||
index_t num_k_mean_var_count_iteration,
|
||||
index_t num_k_block_tile_iteration,
|
||||
index_t k_grid_size,
|
||||
ComputeDataType epsilon,
|
||||
const MeanVarDataType* const p_mean_global,
|
||||
const MeanVarDataType* const p_variance_global,
|
||||
const WorkspaceMeanVarDataType* const p_mean_global,
|
||||
const WorkspaceMeanVarDataType* const p_variance_global,
|
||||
const int32_t* const p_welford_count_global,
|
||||
const XDataType* const __restrict__ p_x_global,
|
||||
const GammaDataType* const __restrict__ p_gamma_global,
|
||||
const BetaDataType* const __restrict__ p_beta_global,
|
||||
YDataType* const __restrict__ p_y_global,
|
||||
SaveMeanInvStdDataType* const __restrict__ p_save_mean_global,
|
||||
SaveMeanInvStdDataType* const __restrict__ p_save_inv_std_global,
|
||||
const YElementwiseOperation y_elementwise_op)
|
||||
{
|
||||
GridwiseWelfordNormalization::Run(mean_var_grid_desc_m_kblock,
|
||||
@@ -78,6 +84,8 @@ kernel_normalizationSplitK2nd(const MeanVarGridDesc_M_KBlock mean_var_grid_desc_
|
||||
gamma_grid_desc_m_k,
|
||||
beta_grid_desc_m_k,
|
||||
y_grid_desc_m_k,
|
||||
save_mean_grid_desc_m,
|
||||
save_inv_std_grid_desc_m,
|
||||
num_k_mean_var_count_iteration,
|
||||
num_k_block_tile_iteration,
|
||||
k_grid_size,
|
||||
@@ -89,6 +97,8 @@ kernel_normalizationSplitK2nd(const MeanVarGridDesc_M_KBlock mean_var_grid_desc_
|
||||
p_gamma_global,
|
||||
p_beta_global,
|
||||
p_y_global,
|
||||
p_save_mean_global,
|
||||
p_save_inv_std_global,
|
||||
y_elementwise_op);
|
||||
};
|
||||
} // namespace ck
|
||||
@@ -107,6 +117,7 @@ template <typename XDataType,
|
||||
typename BetaDataType,
|
||||
typename ComputeDataType,
|
||||
typename YDataType,
|
||||
typename SaveMeanInvStdDataType,
|
||||
typename YElementwiseOperation,
|
||||
index_t Rank,
|
||||
index_t NumReduceDim,
|
||||
@@ -121,17 +132,18 @@ template <typename XDataType,
|
||||
index_t GammaSrcVectorSize,
|
||||
index_t BetaSrcVectorDim,
|
||||
index_t BetaSrcVectorSize,
|
||||
index_t YDstVectorSize>
|
||||
index_t YDstVectorSize,
|
||||
index_t SaveMeanInvStdDstVectorSize>
|
||||
struct DeviceNormalizationSplitKImpl : public DeviceNormalization<XDataType,
|
||||
GammaDataType,
|
||||
BetaDataType,
|
||||
ComputeDataType,
|
||||
YDataType,
|
||||
SaveMeanInvStdDataType,
|
||||
YElementwiseOperation,
|
||||
Rank,
|
||||
NumReduceDim>
|
||||
{
|
||||
using MeanVarDataType = ComputeDataType;
|
||||
using WorkspaceMeanVarDataType = SaveMeanInvStdDataType;
|
||||
|
||||
static_assert(BlockSize == MThreadClusterSize * KThreadClusterSize);
|
||||
static_assert(
|
||||
@@ -144,22 +156,28 @@ struct DeviceNormalizationSplitKImpl : public DeviceNormalization<XDataType,
|
||||
(BetaSrcVectorDim == 1 && KThreadSliceSize % BetaSrcVectorSize == 0)),
|
||||
"Invalid thread slice sizes and/or beta vector sizes configuration, please check!");
|
||||
|
||||
static_assert(MThreadSliceSize % SaveMeanInvStdDstVectorSize == 0,
|
||||
"Invalid thread slice sizes and/or save mean and inverse std vector sizes "
|
||||
"configuration, please check!");
|
||||
|
||||
using PassThrough = tensor_operation::element_wise::PassThrough;
|
||||
|
||||
static constexpr auto I0 = Number<0>{};
|
||||
static constexpr auto I1 = Number<1>{};
|
||||
|
||||
static constexpr index_t NumInvariantDim = Rank - NumReduceDim;
|
||||
static constexpr index_t M_BlockTileSize = MThreadClusterSize * MThreadSliceSize;
|
||||
static constexpr index_t K_BlockTileSize = KThreadClusterSize * KThreadSliceSize;
|
||||
|
||||
static constexpr bool reduceAllDim = (NumInvariantDim == 0);
|
||||
static_assert(!reduceAllDim); // TODO
|
||||
|
||||
static auto MakeSrc2dDescriptor(const std::vector<index_t>& inLengths,
|
||||
const std::vector<index_t>& inStrides,
|
||||
int kBlockSize,
|
||||
int numBlockTileIteration)
|
||||
{
|
||||
constexpr index_t NumInvariantDim = Rank - NumReduceDim;
|
||||
static constexpr index_t numSrcDim = Rank;
|
||||
static constexpr bool reduceAllDim = (NumInvariantDim == 0);
|
||||
|
||||
const auto tupleSrcLengths = make_tuple_from_array(inLengths, Number<numSrcDim>{});
|
||||
const auto tupleSrcStrides = make_tuple_from_array(inStrides, Number<numSrcDim>{});
|
||||
@@ -219,7 +237,7 @@ struct DeviceNormalizationSplitKImpl : public DeviceNormalization<XDataType,
|
||||
};
|
||||
|
||||
template <typename DoPads, index_t MPerTile, index_t KPerTile>
|
||||
static auto MakeMeanVarDescriptor_M_K(index_t M, index_t K)
|
||||
static auto MakeWorkspaceMeanVarDescriptor_M_K(index_t M, index_t K)
|
||||
{
|
||||
const auto grid_desc_m_k =
|
||||
make_naive_tensor_descriptor(make_tuple(M, K), make_tuple(K, I1));
|
||||
@@ -227,26 +245,57 @@ struct DeviceNormalizationSplitKImpl : public DeviceNormalization<XDataType,
|
||||
}
|
||||
|
||||
template <typename DoPads, index_t MPerTile, index_t KPerTile>
|
||||
static auto MakeCountDescriptor_M_K(index_t M, index_t K)
|
||||
static auto MakeWorkspaceCountDescriptor_M_K(index_t M, index_t K)
|
||||
{
|
||||
const auto grid_desc_m_k =
|
||||
make_naive_tensor_descriptor(make_tuple(M, K), make_tuple(I0, I1));
|
||||
return PadTensorDescriptor(grid_desc_m_k, make_tuple(MPerTile, KPerTile), DoPads{});
|
||||
}
|
||||
|
||||
static auto MakeSaveMeanInvStdDescriptor_M(const std::vector<index_t>& lengths,
|
||||
const std::vector<index_t>& strides)
|
||||
{
|
||||
using InvariantDims = typename arithmetic_sequence_gen<0, NumInvariantDim, 1>::type;
|
||||
|
||||
const auto tupleSrcLengths = make_tuple_from_array_and_index_seq(lengths, InvariantDims{});
|
||||
const auto tupleSrcStrides = make_tuple_from_array_and_index_seq(strides, InvariantDims{});
|
||||
|
||||
const auto desc = make_naive_tensor_descriptor(tupleSrcLengths, tupleSrcStrides);
|
||||
|
||||
const auto grid_desc_m =
|
||||
transform_tensor_descriptor(desc,
|
||||
make_tuple(make_merge_transform(tupleSrcLengths)),
|
||||
make_tuple(InvariantDims{}),
|
||||
make_tuple(Sequence<0>{}));
|
||||
|
||||
const auto invariantLength = grid_desc_m.GetLength(Number<0>{});
|
||||
const auto pad_M =
|
||||
math::integer_least_multiple(invariantLength, M_BlockTileSize) - invariantLength;
|
||||
|
||||
auto grid_desc_m_padded = transform_tensor_descriptor(
|
||||
grid_desc_m,
|
||||
make_tuple(make_right_pad_transform(invariantLength, pad_M)),
|
||||
make_tuple(Sequence<0>{}),
|
||||
make_tuple(Sequence<0>{}));
|
||||
|
||||
return grid_desc_m_padded;
|
||||
}
|
||||
|
||||
using SrcGridDesc_M_K = decltype(MakeSrc2dDescriptor({1}, {1}, 1, 1));
|
||||
using Kernel1MeanVarGridDesc_M_KBlock =
|
||||
decltype(MakeMeanVarDescriptor_M_K<Sequence<true, false>, 1, 1>(1, 1));
|
||||
decltype(MakeWorkspaceMeanVarDescriptor_M_K<Sequence<true, false>, 1, 1>(1, 1));
|
||||
|
||||
using Kernel2MeanVarGridDesc_M_KBlock =
|
||||
decltype(MakeMeanVarDescriptor_M_K<Sequence<true, true>, 1, 1>(1, 1));
|
||||
decltype(MakeWorkspaceMeanVarDescriptor_M_K<Sequence<true, true>, 1, 1>(1, 1));
|
||||
|
||||
using Kernel2CountGridDesc_M_KBlock =
|
||||
decltype(MakeCountDescriptor_M_K<Sequence<true, true>, 1, 1>(1, 1));
|
||||
decltype(MakeWorkspaceCountDescriptor_M_K<Sequence<true, true>, 1, 1>(1, 1));
|
||||
|
||||
using SaveMeanInvStdGridDesc_M = decltype(MakeSaveMeanInvStdDescriptor_M({1}, {1}));
|
||||
|
||||
using GridwiseWelford = GridwiseNormalizationSplitK1st<XDataType,
|
||||
ComputeDataType,
|
||||
MeanVarDataType,
|
||||
WorkspaceMeanVarDataType,
|
||||
SrcGridDesc_M_K,
|
||||
Kernel1MeanVarGridDesc_M_KBlock,
|
||||
BlockSize,
|
||||
@@ -258,16 +307,18 @@ struct DeviceNormalizationSplitKImpl : public DeviceNormalization<XDataType,
|
||||
XSrcVectorSize>;
|
||||
|
||||
using GridwiseWelfordNormalization =
|
||||
GridwiseNormalizationSplitK2nd<MeanVarDataType,
|
||||
GridwiseNormalizationSplitK2nd<WorkspaceMeanVarDataType,
|
||||
XDataType,
|
||||
GammaDataType,
|
||||
BetaDataType,
|
||||
YDataType,
|
||||
SaveMeanInvStdDataType,
|
||||
ComputeDataType,
|
||||
YElementwiseOperation,
|
||||
Kernel2MeanVarGridDesc_M_KBlock,
|
||||
Kernel2CountGridDesc_M_KBlock,
|
||||
SrcGridDesc_M_K,
|
||||
SaveMeanInvStdGridDesc_M,
|
||||
BlockSize,
|
||||
MThreadClusterSize,
|
||||
KThreadClusterSize,
|
||||
@@ -280,7 +331,8 @@ struct DeviceNormalizationSplitKImpl : public DeviceNormalization<XDataType,
|
||||
BetaSrcVectorDim,
|
||||
BetaSrcVectorSize,
|
||||
XYVectorDim,
|
||||
YDstVectorSize>;
|
||||
YDstVectorSize,
|
||||
SaveMeanInvStdDstVectorSize>;
|
||||
|
||||
struct Argument : public BaseArgument
|
||||
{
|
||||
@@ -289,17 +341,23 @@ struct DeviceNormalizationSplitKImpl : public DeviceNormalization<XDataType,
|
||||
const std::vector<index_t> gammaStrides,
|
||||
const std::vector<index_t> betaStrides,
|
||||
const std::vector<index_t> yStrides,
|
||||
const std::vector<index_t> saveMeanStrides,
|
||||
const std::vector<index_t> saveInvStdStrides,
|
||||
const std::vector<index_t> reduceDims,
|
||||
YElementwiseOperation y_elementwise_op,
|
||||
double epsilon,
|
||||
const XDataType* p_x,
|
||||
const GammaDataType* p_gamma,
|
||||
const BetaDataType* p_beta,
|
||||
YDataType* p_y)
|
||||
YDataType* p_y,
|
||||
SaveMeanInvStdDataType* p_saveMean,
|
||||
SaveMeanInvStdDataType* p_saveInvStd)
|
||||
: p_x_(p_x),
|
||||
p_gamma_(p_gamma),
|
||||
p_beta_(p_beta),
|
||||
p_y_(p_y),
|
||||
p_saveMean_(p_saveMean),
|
||||
p_saveInvStd_(p_saveInvStd),
|
||||
p_workspace_mean_{nullptr},
|
||||
p_workspace_var_{nullptr},
|
||||
p_workspace_count_{nullptr},
|
||||
@@ -312,6 +370,8 @@ struct DeviceNormalizationSplitKImpl : public DeviceNormalization<XDataType,
|
||||
yStrides_ = shuffle_tensor_dimensions<Rank, NumReduceDim>(yStrides, reduceDims);
|
||||
gammaStrides_ = shuffle_tensor_dimensions<Rank, NumReduceDim>(gammaStrides, reduceDims);
|
||||
betaStrides_ = shuffle_tensor_dimensions<Rank, NumReduceDim>(betaStrides, reduceDims);
|
||||
saveMeanStrides_ = saveMeanStrides;
|
||||
saveInvStdStrides_ = saveInvStdStrides;
|
||||
|
||||
std::tie(MRaw_, KRaw_) = get_2d_lengths<Rank, NumReduceDim>(Lengths_);
|
||||
|
||||
@@ -346,20 +406,28 @@ struct DeviceNormalizationSplitKImpl : public DeviceNormalization<XDataType,
|
||||
y_grid_desc_m_k_ =
|
||||
MakeSrc2dDescriptor(Lengths_, yStrides_, kGridSize_, numBlockTileIteration_);
|
||||
|
||||
save_mean_grid_desc_m_ = MakeSaveMeanInvStdDescriptor_M(Lengths_, saveMeanStrides);
|
||||
save_inv_std_grid_desc_m_ = MakeSaveMeanInvStdDescriptor_M(Lengths_, saveInvStdStrides);
|
||||
|
||||
// We don't need to pad in K dimension for Welford1. Set KPerTile 1.
|
||||
kernel1_mean_var_grid_desc_m_kblock_ =
|
||||
MakeMeanVarDescriptor_M_K<Sequence<true, false>, M_BlockTileSize, 1>(MRaw_,
|
||||
kGridSize_);
|
||||
MakeWorkspaceMeanVarDescriptor_M_K<Sequence<true, false>, M_BlockTileSize, 1>(
|
||||
MRaw_, kGridSize_);
|
||||
|
||||
kernel2_mean_var_grid_desc_m_kblock_ =
|
||||
MakeMeanVarDescriptor_M_K<Sequence<true, true>,
|
||||
M_BlockTileSize,
|
||||
K_MeanVarCountBlockTileSize>(MRaw_, kGridSize_);
|
||||
MakeWorkspaceMeanVarDescriptor_M_K<Sequence<true, true>,
|
||||
M_BlockTileSize,
|
||||
K_MeanVarCountBlockTileSize>(MRaw_, kGridSize_);
|
||||
|
||||
kernel2_count_grid_desc_m_kblock_ =
|
||||
MakeCountDescriptor_M_K<Sequence<true, true>,
|
||||
M_BlockTileSize,
|
||||
K_MeanVarCountBlockTileSize>(MRaw_, kGridSize_);
|
||||
MakeWorkspaceCountDescriptor_M_K<Sequence<true, true>,
|
||||
M_BlockTileSize,
|
||||
K_MeanVarCountBlockTileSize>(MRaw_, kGridSize_);
|
||||
|
||||
if constexpr(NumInvariantDim == 0)
|
||||
invariant_lowest_length_ = 1;
|
||||
else
|
||||
invariant_lowest_length_ = Lengths_[NumInvariantDim - 1];
|
||||
}
|
||||
|
||||
ComputeDataType epsilon_;
|
||||
@@ -368,6 +436,8 @@ struct DeviceNormalizationSplitKImpl : public DeviceNormalization<XDataType,
|
||||
const GammaDataType* p_gamma_;
|
||||
const BetaDataType* p_beta_;
|
||||
YDataType* p_y_;
|
||||
SaveMeanInvStdDataType* p_saveMean_;
|
||||
SaveMeanInvStdDataType* p_saveInvStd_;
|
||||
void* p_workspace_mean_;
|
||||
void* p_workspace_var_;
|
||||
void* p_workspace_count_;
|
||||
@@ -377,6 +447,8 @@ struct DeviceNormalizationSplitKImpl : public DeviceNormalization<XDataType,
|
||||
std::vector<index_t> gammaStrides_;
|
||||
std::vector<index_t> betaStrides_;
|
||||
std::vector<index_t> yStrides_;
|
||||
std::vector<index_t> saveMeanStrides_;
|
||||
std::vector<index_t> saveInvStdStrides_;
|
||||
|
||||
YElementwiseOperation y_elementwise_op_;
|
||||
|
||||
@@ -389,6 +461,8 @@ struct DeviceNormalizationSplitKImpl : public DeviceNormalization<XDataType,
|
||||
SrcGridDesc_M_K gamma_grid_desc_m_k_;
|
||||
SrcGridDesc_M_K beta_grid_desc_m_k_;
|
||||
SrcGridDesc_M_K y_grid_desc_m_k_;
|
||||
SaveMeanInvStdGridDesc_M save_mean_grid_desc_m_;
|
||||
SaveMeanInvStdGridDesc_M save_inv_std_grid_desc_m_;
|
||||
|
||||
Kernel1MeanVarGridDesc_M_KBlock kernel1_mean_var_grid_desc_m_kblock_;
|
||||
Kernel2MeanVarGridDesc_M_KBlock kernel2_mean_var_grid_desc_m_kblock_;
|
||||
@@ -396,6 +470,8 @@ struct DeviceNormalizationSplitKImpl : public DeviceNormalization<XDataType,
|
||||
|
||||
index_t MRaw_; // invarient length
|
||||
index_t KRaw_; // reduce length
|
||||
|
||||
index_t invariant_lowest_length_;
|
||||
};
|
||||
|
||||
struct Invoker : public BaseInvoker
|
||||
@@ -408,60 +484,68 @@ struct DeviceNormalizationSplitKImpl : public DeviceNormalization<XDataType,
|
||||
|
||||
auto kernel1 = kernel_normalizationSplitK1st<GridwiseWelford,
|
||||
XDataType,
|
||||
MeanVarDataType,
|
||||
WorkspaceMeanVarDataType,
|
||||
ComputeDataType,
|
||||
SrcGridDesc_M_K,
|
||||
Kernel1MeanVarGridDesc_M_KBlock>;
|
||||
|
||||
auto kernel2 = kernel_normalizationSplitK2nd<GridwiseWelfordNormalization,
|
||||
MeanVarDataType,
|
||||
WorkspaceMeanVarDataType,
|
||||
XDataType,
|
||||
GammaDataType,
|
||||
BetaDataType,
|
||||
YDataType,
|
||||
SaveMeanInvStdDataType,
|
||||
ComputeDataType,
|
||||
YElementwiseOperation,
|
||||
Kernel2MeanVarGridDesc_M_KBlock,
|
||||
Kernel2CountGridDesc_M_KBlock,
|
||||
SrcGridDesc_M_K>;
|
||||
SrcGridDesc_M_K,
|
||||
SaveMeanInvStdGridDesc_M>;
|
||||
|
||||
float avg_time = 0;
|
||||
avg_time += launch_and_time_kernel(stream_config,
|
||||
kernel1,
|
||||
dim3(arg.gridSize_),
|
||||
dim3(BlockSize),
|
||||
0,
|
||||
arg.x_grid_desc_m_k_,
|
||||
arg.kernel1_mean_var_grid_desc_m_kblock_,
|
||||
arg.numBlockTileIteration_,
|
||||
arg.p_x_,
|
||||
static_cast<MeanVarDataType*>(arg.p_workspace_mean_),
|
||||
static_cast<MeanVarDataType*>(arg.p_workspace_var_),
|
||||
static_cast<int32_t*>(arg.p_workspace_count_));
|
||||
avg_time += launch_and_time_kernel(
|
||||
stream_config,
|
||||
kernel1,
|
||||
dim3(arg.gridSize_),
|
||||
dim3(BlockSize),
|
||||
0,
|
||||
arg.x_grid_desc_m_k_,
|
||||
arg.kernel1_mean_var_grid_desc_m_kblock_,
|
||||
arg.numBlockTileIteration_,
|
||||
arg.p_x_,
|
||||
static_cast<WorkspaceMeanVarDataType*>(arg.p_workspace_mean_),
|
||||
static_cast<WorkspaceMeanVarDataType*>(arg.p_workspace_var_),
|
||||
static_cast<int32_t*>(arg.p_workspace_count_));
|
||||
|
||||
avg_time += launch_and_time_kernel(stream_config,
|
||||
kernel2,
|
||||
dim3(arg.gridSize_),
|
||||
dim3(BlockSize),
|
||||
0,
|
||||
arg.kernel2_mean_var_grid_desc_m_kblock_,
|
||||
arg.kernel2_count_grid_desc_m_kblock_,
|
||||
arg.x_grid_desc_m_k_,
|
||||
arg.gamma_grid_desc_m_k_,
|
||||
arg.beta_grid_desc_m_k_,
|
||||
arg.y_grid_desc_m_k_,
|
||||
arg.numMeanVarCountIteration_,
|
||||
arg.numBlockTileIteration_,
|
||||
arg.kGridSize_,
|
||||
arg.epsilon_,
|
||||
static_cast<MeanVarDataType*>(arg.p_workspace_mean_),
|
||||
static_cast<MeanVarDataType*>(arg.p_workspace_var_),
|
||||
static_cast<int32_t*>(arg.p_workspace_count_),
|
||||
arg.p_x_,
|
||||
arg.p_gamma_,
|
||||
arg.p_beta_,
|
||||
arg.p_y_,
|
||||
arg.y_elementwise_op_);
|
||||
avg_time += launch_and_time_kernel(
|
||||
stream_config,
|
||||
kernel2,
|
||||
dim3(arg.gridSize_),
|
||||
dim3(BlockSize),
|
||||
0,
|
||||
arg.kernel2_mean_var_grid_desc_m_kblock_,
|
||||
arg.kernel2_count_grid_desc_m_kblock_,
|
||||
arg.x_grid_desc_m_k_,
|
||||
arg.gamma_grid_desc_m_k_,
|
||||
arg.beta_grid_desc_m_k_,
|
||||
arg.y_grid_desc_m_k_,
|
||||
arg.save_mean_grid_desc_m_,
|
||||
arg.save_inv_std_grid_desc_m_,
|
||||
arg.numMeanVarCountIteration_,
|
||||
arg.numBlockTileIteration_,
|
||||
arg.kGridSize_,
|
||||
arg.epsilon_,
|
||||
static_cast<const WorkspaceMeanVarDataType*>(arg.p_workspace_mean_),
|
||||
static_cast<const WorkspaceMeanVarDataType*>(arg.p_workspace_var_),
|
||||
static_cast<const int32_t*>(arg.p_workspace_count_),
|
||||
arg.p_x_,
|
||||
arg.p_gamma_,
|
||||
arg.p_beta_,
|
||||
arg.p_y_,
|
||||
arg.p_saveMean_,
|
||||
arg.p_saveInvStd_,
|
||||
arg.y_elementwise_op_);
|
||||
|
||||
return avg_time;
|
||||
};
|
||||
@@ -482,10 +566,10 @@ struct DeviceNormalizationSplitKImpl : public DeviceNormalization<XDataType,
|
||||
int welford_size = pArg_->MRaw_ * pArg_->kGridSize_;
|
||||
|
||||
// workspace for welford intermediate mean
|
||||
workspace_size += welford_size * sizeof(MeanVarDataType) + 64;
|
||||
workspace_size += welford_size * sizeof(WorkspaceMeanVarDataType) + 64;
|
||||
|
||||
// workspace for welford intermediate variance
|
||||
workspace_size += welford_size * sizeof(MeanVarDataType) + 64;
|
||||
workspace_size += welford_size * sizeof(WorkspaceMeanVarDataType) + 64;
|
||||
|
||||
// workspace for welford intermediate count
|
||||
workspace_size += pArg_->kGridSize_ * sizeof(int32_t) + 64;
|
||||
@@ -504,13 +588,13 @@ struct DeviceNormalizationSplitKImpl : public DeviceNormalization<XDataType,
|
||||
// setup buffer used for intermediate welford mean
|
||||
pArg_->p_workspace_mean_ = static_cast<char*>(pArg_->p_workspace_);
|
||||
|
||||
index_t mean_space_sz = welford_size * sizeof(MeanVarDataType);
|
||||
index_t mean_space_sz = welford_size * sizeof(WorkspaceMeanVarDataType);
|
||||
mean_space_sz = math::integer_least_multiple(mean_space_sz, 64);
|
||||
|
||||
// setup buffer used for intermediate welford varirance
|
||||
pArg_->p_workspace_var_ = reinterpret_cast<char*>(pArg_->p_workspace_mean_) + mean_space_sz;
|
||||
|
||||
index_t variance_space_sz = welford_size * sizeof(MeanVarDataType);
|
||||
index_t variance_space_sz = welford_size * sizeof(WorkspaceMeanVarDataType);
|
||||
variance_space_sz = math::integer_least_multiple(variance_space_sz, 64);
|
||||
|
||||
// setup buffer used for intermediate welford count
|
||||
@@ -522,8 +606,6 @@ struct DeviceNormalizationSplitKImpl : public DeviceNormalization<XDataType,
|
||||
{
|
||||
const Argument* p_arg_ = dynamic_cast<const Argument*>(p_arg);
|
||||
|
||||
constexpr index_t NumInvariantDim = Rank - NumReduceDim;
|
||||
|
||||
if constexpr(XYVectorDim == 0)
|
||||
{
|
||||
if constexpr(NumInvariantDim == 0)
|
||||
@@ -535,10 +617,10 @@ struct DeviceNormalizationSplitKImpl : public DeviceNormalization<XDataType,
|
||||
if(p_arg_->xStrides_[NumInvariantDim - 1] != 1)
|
||||
return false;
|
||||
|
||||
if(p_arg_->invariant_lowest_length % XSrcVectorSize != 0)
|
||||
if(p_arg_->invariant_lowest_length_ % XSrcVectorSize != 0)
|
||||
return false;
|
||||
|
||||
if(p_arg_->invariant_lowest_length % YDstVectorSize != 0)
|
||||
if(p_arg_->invariant_lowest_length_ % YDstVectorSize != 0)
|
||||
return false;
|
||||
};
|
||||
}
|
||||
@@ -578,7 +660,7 @@ struct DeviceNormalizationSplitKImpl : public DeviceNormalization<XDataType,
|
||||
if(p_arg_->betaStrides_[NumInvariantDim - 1] != 1)
|
||||
return false;
|
||||
|
||||
if(p_arg_->invariant_lowest_length % BetaSrcVectorSize != 0)
|
||||
if(p_arg_->invariant_lowest_length_ % BetaSrcVectorSize != 0)
|
||||
return false;
|
||||
}
|
||||
else // if fastest dim is reduced
|
||||
@@ -593,6 +675,9 @@ struct DeviceNormalizationSplitKImpl : public DeviceNormalization<XDataType,
|
||||
if(p_arg_->kGridSize_ <= 1)
|
||||
return false;
|
||||
|
||||
if(p_arg_->invariant_lowest_length_ % SaveMeanInvStdDstVectorSize != 0)
|
||||
return false;
|
||||
|
||||
return true;
|
||||
};
|
||||
|
||||
@@ -602,6 +687,8 @@ struct DeviceNormalizationSplitKImpl : public DeviceNormalization<XDataType,
|
||||
const std::vector<index_t> gammaStrides,
|
||||
const std::vector<index_t> betaStrides,
|
||||
const std::vector<index_t> yStrides,
|
||||
const std::vector<index_t> saveMeanStrides,
|
||||
const std::vector<index_t> saveInvStdStrides,
|
||||
const std::vector<index_t> reduceDims,
|
||||
double epsilon,
|
||||
const void* p_x,
|
||||
@@ -609,27 +696,30 @@ struct DeviceNormalizationSplitKImpl : public DeviceNormalization<XDataType,
|
||||
const void* p_beta,
|
||||
void* p_y,
|
||||
void* p_saveMean,
|
||||
void* p_saveInvVar,
|
||||
void* p_saveInvStd,
|
||||
YElementwiseOperation y_elementwise_op) override
|
||||
{
|
||||
// TODO
|
||||
// Optional cache of the intermediate results (mean and InvVariance) during the
|
||||
// forward pass could speedup in the backward
|
||||
ignore = p_saveMean;
|
||||
ignore = p_saveInvVar;
|
||||
if(lengths.size() != Rank || xStrides.size() != Rank || gammaStrides.size() != Rank ||
|
||||
betaStrides.size() != Rank || yStrides.size() != Rank ||
|
||||
saveMeanStrides.size() != NumInvariantDim || saveInvStdStrides.size() != NumInvariantDim)
|
||||
throw std::runtime_error("dimension is incorrect");
|
||||
|
||||
return std::make_unique<Argument>(lengths,
|
||||
xStrides,
|
||||
gammaStrides,
|
||||
betaStrides,
|
||||
yStrides,
|
||||
saveMeanStrides,
|
||||
saveInvStdStrides,
|
||||
reduceDims,
|
||||
y_elementwise_op,
|
||||
epsilon,
|
||||
static_cast<const XDataType*>(p_x),
|
||||
static_cast<const GammaDataType*>(p_gamma),
|
||||
static_cast<const BetaDataType*>(p_beta),
|
||||
static_cast<YDataType*>(p_y));
|
||||
static_cast<YDataType*>(p_y),
|
||||
static_cast<SaveMeanInvStdDataType*>(p_saveMean),
|
||||
static_cast<SaveMeanInvStdDataType*>(p_saveInvStd));
|
||||
};
|
||||
|
||||
std::unique_ptr<BaseInvoker> MakeInvokerPointer() override
|
||||
|
||||
Reference in New Issue
Block a user