first commit - ReverseSequence - GPU

Added additional comments + added test case + lint

linting + fixed dyn cast to static cast

comment add

MindTester case fix - 0 edge case + improved comments

addressing some comments + malloc remove from kernel

Added test case for 0 edge case testing
This commit is contained in:
danishfarid 2021-05-31 13:50:52 -04:00
parent 05bbd1d8fa
commit e4d33f1195
6 changed files with 522 additions and 1 deletions

View File

@ -0,0 +1,95 @@
/**
* Copyright 2021 Huawei Technologies Co., Ltd
*
* Licensed under the Apache License, Version 2.0 (the "License");
* you may not use this file except in compliance with the License.
* You may obtain a copy of the License at
*
* http://www.apache.org/licenses/LICENSE-2.0
*
* Unless required by applicable law or agreed to in writing, software
* distributed under the License is distributed on an "AS IS" BASIS,
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
* See the License for the specific language governing permissions and
* limitations under the License.
*/
#include "backend/kernel_compiler/gpu/arrays/reverse_sequence_gpu_kernel.h"
namespace mindspore {
namespace kernel {
// int8
MS_REG_GPU_KERNEL_TWO(
ReverseSequence,
KernelAttr().AddInputAttr(kNumberTypeInt8).AddInputAttr(kNumberTypeInt32).AddOutputAttr(kNumberTypeInt8),
ReverseSequenceGpuFwdKernel, int8_t, int)
MS_REG_GPU_KERNEL_TWO(
ReverseSequence,
KernelAttr().AddInputAttr(kNumberTypeInt8).AddInputAttr(kNumberTypeInt64).AddOutputAttr(kNumberTypeInt8),
ReverseSequenceGpuFwdKernel, int8_t, int64_t)
// int16
MS_REG_GPU_KERNEL_TWO(
ReverseSequence,
KernelAttr().AddInputAttr(kNumberTypeInt16).AddInputAttr(kNumberTypeInt32).AddOutputAttr(kNumberTypeInt16),
ReverseSequenceGpuFwdKernel, int16_t, int)
MS_REG_GPU_KERNEL_TWO(
ReverseSequence,
KernelAttr().AddInputAttr(kNumberTypeInt16).AddInputAttr(kNumberTypeInt64).AddOutputAttr(kNumberTypeInt16),
ReverseSequenceGpuFwdKernel, int16_t, int64_t)
// int32
MS_REG_GPU_KERNEL_TWO(
ReverseSequence,
KernelAttr().AddInputAttr(kNumberTypeInt32).AddInputAttr(kNumberTypeInt32).AddOutputAttr(kNumberTypeInt32),
ReverseSequenceGpuFwdKernel, int, int)
MS_REG_GPU_KERNEL_TWO(
ReverseSequence,
KernelAttr().AddInputAttr(kNumberTypeInt32).AddInputAttr(kNumberTypeInt64).AddOutputAttr(kNumberTypeInt32),
ReverseSequenceGpuFwdKernel, int, int64_t)
// int64
MS_REG_GPU_KERNEL_TWO(
ReverseSequence,
KernelAttr().AddInputAttr(kNumberTypeInt64).AddInputAttr(kNumberTypeInt32).AddOutputAttr(kNumberTypeInt64),
ReverseSequenceGpuFwdKernel, int64_t, int)
MS_REG_GPU_KERNEL_TWO(
ReverseSequence,
KernelAttr().AddInputAttr(kNumberTypeInt64).AddInputAttr(kNumberTypeInt64).AddOutputAttr(kNumberTypeInt64),
ReverseSequenceGpuFwdKernel, int64_t, int64_t)
// float16
MS_REG_GPU_KERNEL_TWO(
ReverseSequence,
KernelAttr().AddInputAttr(kNumberTypeFloat16).AddInputAttr(kNumberTypeInt32).AddOutputAttr(kNumberTypeFloat16),
ReverseSequenceGpuFwdKernel, half, int)
MS_REG_GPU_KERNEL_TWO(
ReverseSequence,
KernelAttr().AddInputAttr(kNumberTypeFloat16).AddInputAttr(kNumberTypeInt64).AddOutputAttr(kNumberTypeFloat16),
ReverseSequenceGpuFwdKernel, half, int64_t)
// float32
MS_REG_GPU_KERNEL_TWO(
ReverseSequence,
KernelAttr().AddInputAttr(kNumberTypeFloat32).AddInputAttr(kNumberTypeInt32).AddOutputAttr(kNumberTypeFloat32),
ReverseSequenceGpuFwdKernel, float, int)
MS_REG_GPU_KERNEL_TWO(
ReverseSequence,
KernelAttr().AddInputAttr(kNumberTypeFloat32).AddInputAttr(kNumberTypeInt64).AddOutputAttr(kNumberTypeFloat32),
ReverseSequenceGpuFwdKernel, float, int64_t)
// float64
MS_REG_GPU_KERNEL_TWO(
ReverseSequence,
KernelAttr().AddInputAttr(kNumberTypeFloat64).AddInputAttr(kNumberTypeInt32).AddOutputAttr(kNumberTypeFloat64),
ReverseSequenceGpuFwdKernel, double, int)
MS_REG_GPU_KERNEL_TWO(
ReverseSequence,
KernelAttr().AddInputAttr(kNumberTypeFloat64).AddInputAttr(kNumberTypeInt64).AddOutputAttr(kNumberTypeFloat64),
ReverseSequenceGpuFwdKernel, double, int64_t)
// bool
MS_REG_GPU_KERNEL_TWO(
ReverseSequence,
KernelAttr().AddInputAttr(kNumberTypeBool).AddInputAttr(kNumberTypeInt32).AddOutputAttr(kNumberTypeBool),
ReverseSequenceGpuFwdKernel, bool, int)
MS_REG_GPU_KERNEL_TWO(
ReverseSequence,
KernelAttr().AddInputAttr(kNumberTypeBool).AddInputAttr(kNumberTypeInt64).AddOutputAttr(kNumberTypeBool),
ReverseSequenceGpuFwdKernel, bool, int64_t)
} // namespace kernel
} // namespace mindspore

View File

@ -0,0 +1,130 @@
/**
* Copyright 2021 Huawei Technologies Co., Ltd
*
* Licensed under the Apache License, Version 2.0 (the "License");
* you may not use this file except in compliance with the License.
* You may obtain a copy of the License at
*
* http://www.apache.org/licenses/LICENSE-2.0
*
* Unless required by applicable law or agreed to in writing, software
* distributed under the License is distributed on an "AS IS" BASIS,
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
* See the License for the specific language governing permissions and
* limitations under the License.
*/
#ifndef MINDSPORE_CCSRC_BACKEND_KERNEL_COMPILER_GPU_ARRAYS_REVERSE_SEQUENCE_GPU_KERNEL_H_
#define MINDSPORE_CCSRC_BACKEND_KERNEL_COMPILER_GPU_ARRAYS_REVERSE_SEQUENCE_GPU_KERNEL_H_
#include <vector>
#include <memory>
#include <iostream>
#include "backend/kernel_compiler/gpu/gpu_kernel.h"
#include "backend/kernel_compiler/gpu/gpu_kernel_factory.h"
#include "runtime/device/gpu/cuda_common.h"
#include "backend/kernel_compiler/gpu/cuda_impl/reverse_sequence_impl.cuh"
#include "backend/kernel_compiler/gpu/kernel_constants.h"
namespace mindspore {
namespace kernel {
template <typename T, typename S>
class ReverseSequenceGpuFwdKernel : public GpuKernel {
public:
ReverseSequenceGpuFwdKernel()
: shape_size_(0),
input_size_(0),
batch_dim_(0),
seq_dim_(0),
seq_len_size_(0),
total_index_dim_(0),
output_size_(0),
workspace_size_(0) {}
~ReverseSequenceGpuFwdKernel() override = default;
const std::vector<size_t> &GetInputSizeList() const override { return input_size_list_; }
const std::vector<size_t> &GetOutputSizeList() const override { return output_size_list_; }
const std::vector<size_t> &GetWorkspaceSizeList() const override { return workspace_size_list_; }
bool Launch(const std::vector<AddressPtr> &inputs, const std::vector<AddressPtr> &workspace,
const std::vector<AddressPtr> &outputs, void *stream_ptr) override {
T *input = GetDeviceAddress<T>(inputs, 0);
S *seq_len = GetDeviceAddress<S>(inputs, 1);
size_t *input_shape_ptr = GetDeviceAddress<size_t>(workspace, 0);
size_t *input_cum_shape_ptr = GetDeviceAddress<size_t>(workspace, 1);
size_t *cur_pos_arr = GetDeviceAddress<size_t>(workspace, 2);
T *output = GetDeviceAddress<T>(outputs, 0);
CHECK_CUDA_RET_WITH_EXCEPT(kernel_node_,
cudaMemcpyAsync(input_shape_ptr, &input_shape_[0], input_shape_.size() * sizeof(size_t),
cudaMemcpyHostToDevice, reinterpret_cast<cudaStream_t>(stream_ptr)),
"cudaMemcpyAsync input_shape_ failed");
CalReverseSequence(input_size_, input, seq_len, batch_dim_, seq_dim_, cur_pos_arr, input_shape_ptr,
input_cum_shape_ptr, shape_size_, output, reinterpret_cast<cudaStream_t>(stream_ptr));
return true;
}
bool Init(const CNodePtr &kernel_node) override {
batch_dim_ = GetAttr<int64_t>(kernel_node, "batch_dim");
seq_dim_ = GetAttr<int64_t>(kernel_node, "seq_dim");
size_t input_num = AnfAlgo::GetInputTensorNum(kernel_node);
if (input_num != 2) {
MS_LOG(ERROR) << "Input number is " << input_num << ", but ReverseSequence needs 2 input.";
return false;
}
size_t output_num = AnfAlgo::GetOutputTensorNum(kernel_node);
if (output_num != 1) {
MS_LOG(ERROR) << "Output number is " << output_num << ", but ReverseSequence needs 1 output.";
return false;
}
input_shape_ = AnfAlgo::GetPrevNodeOutputInferShape(kernel_node, 0);
if (CHECK_NULL_INPUT(input_shape_)) {
MS_LOG(WARNING) << "ReverseSequence input is null";
InitSizeLists();
return true;
}
input_size_ = 1;
shape_size_ = input_shape_.size(); // required for calls
for (size_t i = 0; i < shape_size_; i++) {
input_size_ *= input_shape_[i];
}
// get seq len shape
auto seq_len_shape = AnfAlgo::GetPrevNodeOutputInferShape(kernel_node, 1);
if (CHECK_NULL_INPUT(seq_len_shape)) {
MS_LOG(WARNING) << "ReverseSequence seq lengths input is null";
InitSizeLists();
return true;
}
seq_len_size_ = seq_len_shape.size();
output_size_ = input_size_; // size does not change
// Allocate workspace memory to use for storing indices for each thread to compute with
size_t total_threads = GET_BLOCKS(input_size_) * GET_THREADS;
total_index_dim_ = total_threads * shape_size_;
InitSizeLists();
return true;
}
protected:
void InitSizeLists() override {
input_size_list_.push_back(input_size_ * sizeof(T));
input_size_list_.push_back(seq_len_size_ * sizeof(S));
workspace_size_list_.push_back(shape_size_ * sizeof(size_t)); // input_shape
workspace_size_list_.push_back(shape_size_ * sizeof(size_t)); // cumulative shape
workspace_size_list_.push_back(total_index_dim_ * sizeof(size_t)); // scratch memory for holding indices per thread
output_size_list_.push_back(output_size_ * sizeof(T));
}
private:
size_t shape_size_;
size_t input_size_;
int64_t batch_dim_;
int64_t seq_dim_;
size_t seq_len_size_;
size_t total_index_dim_;
size_t output_size_;
size_t workspace_size_;
std::vector<size_t> input_shape_;
std::vector<size_t> input_size_list_;
std::vector<size_t> output_size_list_;
std::vector<size_t> workspace_size_list_;
};
} // namespace kernel
} // namespace mindspore
#endif // MINDSPORE_CCSRC_BACKEND_KERNEL_COMPILER_GPU_ARRAYS_REVERSE_SEQUENCE_GPU_KERNEL_H_

View File

@ -0,0 +1,159 @@
/**
* Copyright 2021 Huawei Technologies Co., Ltd
*
* Licensed under the Apache License, Version 2.0 (the "License");
* you may not use this file except in compliance with the License.
* You may obtain a copy of the License at
*
* http://www.apache.org/licenses/LICENSE-2.0
*
* Unless required by applicable law or agreed to in writing, software
* distributed under the License is distributed on an "AS IS" BASIS,
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
* See the License for the specific language governing permissions and
* limitations under the License.
*/
#include <assert.h>
#include <stdio.h>
#include <stdint.h>
#include "backend/kernel_compiler/gpu/cuda_impl/reverse_sequence_impl.cuh"
// Util function to convert a 1D input array index to an N-D positional index
// Required since GPU iterates over all values in an ND array as a 1D array
__inline__ __device__ void IdxToPos(size_t idx, size_t *pos, size_t cur_thread_idx, size_t *cum_shape,
size_t shape_size) {
size_t rem_val = idx;
for (int i = 0; i < shape_size; i++) {
pos[cur_thread_idx + i] = rem_val / cum_shape[i];
rem_val = rem_val % cum_shape[i];
}
return;
}
// Util function to convert a N-D positonal index to a 1D index
__inline__ __device__ size_t PosToIdx(size_t *pos, size_t cur_thread_idx, size_t *cum_shape, size_t shape_size) {
size_t idx = 0;
for (int i = 0; i < shape_size; i++) {
idx = idx + (pos[cur_thread_idx + i] * cum_shape[i]);
}
return idx;
}
// CumShape takes Shape: (2,2,5) => cumShape (10,5,1) which informs how many values
// each dimension will represent. Required for converting 1d index to positional vector.
// In this example 10 in dim 0 means, an increase of 1 in this dim leads to another 10 values
// in the overall array
__global__ void ComputeCumShape(const size_t *input_shape_ptr, size_t *input_shape_cum_ptr, size_t shape_size) {
int cur_val = 1;
for (int i = shape_size - 1; i >= 0; i--) {
// iterate list in reverse and cummulatively build shape
input_shape_cum_ptr[i] = cur_val;
cur_val = cur_val * input_shape_ptr[i];
}
return;
}
template <typename T, typename S>
__global__ void ReverseSequence(const size_t size, const T *input, const S *seq_len, const int64_t batch_dim,
const int64_t seq_dim, size_t *cur_pos_arr, const size_t *input_shape_ptr,
size_t *input_shape_cum_ptr, size_t shape_size, T *output) {
// calculate which thread this is out of total across all blocks for accessing respective cur_pos_arr memory
size_t cur_thread_idx = (blockIdx.x * blockDim.x) + threadIdx.x;
cur_thread_idx = cur_thread_idx * shape_size;
size_t cur_slice = 0; // current slice as split by the batch_dim
size_t cur_slice_seq_len = 0; // reverse seq length for this slice as provided by user
size_t new_idx = 0; // calculate corresponding reverse element from input
for (size_t idx = blockIdx.x * blockDim.x + threadIdx.x; idx < size; idx += blockDim.x * gridDim.x) {
IdxToPos(idx, cur_pos_arr, cur_thread_idx, input_shape_cum_ptr, shape_size);
cur_slice = cur_pos_arr[cur_thread_idx + batch_dim]; // all accesses to cur_pos_arr have to be adjusted per thread
cur_slice_seq_len = seq_len[cur_slice];
if (cur_slice_seq_len == 0) { // adjust length to 1 if 0 provided, same result in both cases
cur_slice_seq_len = 1;
}
if (cur_pos_arr[cur_thread_idx + seq_dim] > (cur_slice_seq_len - 1)) { // check if within range
// copy value directly and continue - outside of reversal range
output[idx] = input[idx];
continue;
}
// find corresponding reverse element in input
cur_pos_arr[cur_thread_idx + seq_dim] =
(cur_slice_seq_len - 1) - cur_pos_arr[cur_thread_idx + seq_dim]; // adjust position to target
new_idx = PosToIdx(cur_pos_arr, cur_thread_idx, input_shape_cum_ptr, shape_size); // get the updated index
output[idx] = input[new_idx];
}
return;
}
template <typename T, typename S>
void CalReverseSequence(const size_t size, const T *input, const S *seq_len, const int64_t batch_dim,
const int64_t seq_dim, size_t *cur_pos_arr, const size_t *input_shape_ptr,
size_t *input_shape_cum_ptr, size_t shape_size, T *output, cudaStream_t cuda_stream) {
ComputeCumShape<<<1, 1, 0, cuda_stream>>>(input_shape_ptr, input_shape_cum_ptr, shape_size);
ReverseSequence<<<GET_BLOCKS(size), GET_THREADS, 0, cuda_stream>>>(
size, input, seq_len, batch_dim, seq_dim, cur_pos_arr, input_shape_ptr, input_shape_cum_ptr, shape_size, output);
return;
}
template void CalReverseSequence<int8_t, int>(const size_t size, const int8_t *input, const int *seq_len,
const int64_t batch_dim, const int64_t seq_dim, size_t *cur_pos_arr,
const size_t *input_shape_ptr, size_t *intput_shape_cum_ptr,
size_t shape_size, int8_t *output, cudaStream_t cuda_stream);
template void CalReverseSequence<int8_t, int64_t>(const size_t size, const int8_t *input, const int64_t *seq_len,
const int64_t batch_dim, const int64_t seq_dim, size_t *cur_pos_arr,
const size_t *input_shape_ptr, size_t *intput_shape_cum_ptr,
size_t shape_size, int8_t *output, cudaStream_t cuda_stream);
template void CalReverseSequence<int16_t, int>(const size_t size, const int16_t *input, const int *seq_len,
const int64_t batch_dim, const int64_t seq_dim, size_t *cur_pos_arr,
const size_t *input_shape_ptr, size_t *intput_shape_cum_ptr,
size_t shape_size, int16_t *output, cudaStream_t cuda_stream);
template void CalReverseSequence<int16_t, int64_t>(const size_t size, const int16_t *input, const int64_t *seq_len,
const int64_t batch_dim, const int64_t seq_dim, size_t *cur_pos_arr,
const size_t *input_shape_ptr, size_t *intput_shape_cum_ptr,
size_t shape_size, int16_t *output, cudaStream_t cuda_stream);
template void CalReverseSequence<int, int>(const size_t size, const int *input, const int *seq_len,
const int64_t batch_dim, const int64_t seq_dim, size_t *cur_pos_arr,
const size_t *input_shape_ptr, size_t *intput_shape_cum_ptr,
size_t shape_size, int *output, cudaStream_t cuda_stream);
template void CalReverseSequence<int, int64_t>(const size_t size, const int *input, const int64_t *seq_len,
const int64_t batch_dim, const int64_t seq_dim, size_t *cur_pos_arr,
const size_t *input_shape_ptr, size_t *intput_shape_cum_ptr,
size_t shape_size, int *output, cudaStream_t cuda_stream);
template void CalReverseSequence<int64_t, int>(const size_t size, const int64_t *input, const int *seq_len,
const int64_t batch_dim, const int64_t seq_dim, size_t *cur_pos_arr,
const size_t *input_shape_ptr, size_t *intput_shape_cum_ptr,
size_t shape_size, int64_t *output, cudaStream_t cuda_stream);
template void CalReverseSequence<int64_t, int64_t>(const size_t size, const int64_t *input, const int64_t *seq_len,
const int64_t batch_dim, const int64_t seq_dim, size_t *cur_pos_arr,
const size_t *input_shape_ptr, size_t *intput_shape_cum_ptr,
size_t shape_size, int64_t *output, cudaStream_t cuda_stream);
template void CalReverseSequence<half, int>(const size_t size, const half *input, const int *seq_len,
const int64_t batch_dim, const int64_t seq_dim, size_t *cur_pos_arr,
const size_t *input_shape_ptr, size_t *intput_shape_cum_ptr,
size_t shape_size, half *output, cudaStream_t cuda_stream);
template void CalReverseSequence<half, int64_t>(const size_t size, const half *input, const int64_t *seq_len,
const int64_t batch_dim, const int64_t seq_dim, size_t *cur_pos_arr,
const size_t *input_shape_ptr, size_t *intput_shape_cum_ptr,
size_t shape_size, half *output, cudaStream_t cuda_stream);
template void CalReverseSequence<float, int>(const size_t size, const float *input, const int *seq_len,
const int64_t batch_dim, const int64_t seq_dim, size_t *cur_pos_arr,
const size_t *input_shape_ptr, size_t *intput_shape_cum_ptr,
size_t shape_size, float *output, cudaStream_t cuda_stream);
template void CalReverseSequence<float, int64_t>(const size_t size, const float *input, const int64_t *seq_len,
const int64_t batch_dim, const int64_t seq_dim, size_t *cur_pos_arr,
const size_t *input_shape_ptr, size_t *intput_shape_cum_ptr,
size_t shape_size, float *output, cudaStream_t cuda_stream);
template void CalReverseSequence<double, int>(const size_t size, const double *input, const int *seq_len,
const int64_t batch_dim, const int64_t seq_dim, size_t *cur_pos_arr,
const size_t *input_shape_ptr, size_t *intput_shape_cum_ptr,
size_t shape_size, double *output, cudaStream_t cuda_stream);
template void CalReverseSequence<double, int64_t>(const size_t size, const double *input, const int64_t *seq_len,
const int64_t batch_dim, const int64_t seq_dim, size_t *cur_pos_arr,
const size_t *input_shape_ptr, size_t *intput_shape_cum_ptr,
size_t shape_size, double *output, cudaStream_t cuda_stream);
template void CalReverseSequence<bool, int>(const size_t size, const bool *input, const int *seq_len,
const int64_t batch_dim, const int64_t seq_dim, size_t *cur_pos_arr,
const size_t *input_shape_ptr, size_t *intput_shape_cum_ptr,
size_t shape_size, bool *output, cudaStream_t cuda_stream);
template void CalReverseSequence<bool, int64_t>(const size_t size, const bool *input, const int64_t *seq_len,
const int64_t batch_dim, const int64_t seq_dim, size_t *cur_pos_arr,
const size_t *input_shape_ptr, size_t *intput_shape_cum_ptr,
size_t shape_size, bool *output, cudaStream_t cuda_stream);

View File

@ -0,0 +1,27 @@
/**
* Copyright 2021 Huawei Technologies Co., Ltd
*
* Licensed under the Apache License, Version 2.0 (the "License");
* you may not use this file except in compliance with the License.
* You may obtain a copy of the License at
*
* http://www.apache.org/licenses/LICENSE-2.0
*
* Unless required by applicable law or agreed to in writing, software
* distributed under the License is distributed on an "AS IS" BASIS,
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
* See the License for the specific language governing permissions and
* limitations under the License.
*/
#ifndef MINDSPORE_CCSRC_BACKEND_KERNEL_COMPILER_GPU_CUDA_IMPL_REVERSE_SEQUENCE_IMPL_H_
#define MINDSPORE_CCSRC_BACKEND_KERNEL_COMPILER_GPU_CUDA_IMPL_REVERSE_SEQUENCE_IMPL_H_
#include <cuda_runtime.h>
#include "runtime/device/gpu/cuda_common.h"
template <typename T, typename S>
void CalReverseSequence(const size_t size, const T *input, const S *seq_len, const int64_t batch_dim,
const int64_t seq_dim, size_t *cur_pos_arr, const size_t *input_shape_ptr,
size_t *intput_shape_cum_ptr, size_t shape_size, T *output, cudaStream_t cuda_stream);
#endif // MINDSPORE_CCSRC_BACKEND_KERNEL_COMPILER_GPU_CUDA_IMPL_REVERSE_SEQUENCE_IMPL_H_

View File

@ -4814,7 +4814,7 @@ class ReverseSequence(PrimitiveWithInfer):
TypeError: If `seq_dim` or `batch_dim` is not an int. TypeError: If `seq_dim` or `batch_dim` is not an int.
Supported Platforms: Supported Platforms:
``Ascend`` ``Ascend`` ``GPU``
Examples: Examples:
>>> x = Tensor(np.array([[1, 2, 3], [4, 5, 6], [7, 8, 9]]), mindspore.float32) >>> x = Tensor(np.array([[1, 2, 3], [4, 5, 6], [7, 8, 9]]), mindspore.float32)

View File

@ -0,0 +1,110 @@
import pytest
import numpy as np
import mindspore.context as context
import mindspore.nn as nn
from mindspore import Tensor
from mindspore.common.api import ms_function
from mindspore.ops import operations as P
class Net(nn.Cell):
def __init__(self, seq_dim, batch_dim):
super(Net, self).__init__()
self.reverse_sequence = P.ReverseSequence(
seq_dim=seq_dim, batch_dim=batch_dim)
@ms_function
def construct(self, x, seq_lengths):
return self.reverse_sequence(x, seq_lengths)
@pytest.mark.level0
@pytest.mark.platform_x86_gpu_training
@pytest.mark.env_onecard
def test_net_int8():
context.set_context(mode=context.GRAPH_MODE, device_target="GPU")
x = np.array([[1, 2, 3], [4, 5, 6], [7, 8, 9]]).astype(np.int8)
seq_lengths = np.array([1, 2, 3]).astype(np.int32)
seq_dim = 0
batch_dim = 1
net = Net(seq_dim, batch_dim)
output = net(Tensor(x), Tensor(seq_lengths))
expected = np.array([[1, 5, 9], [4, 2, 6], [7, 8, 3]]).astype(np.int8)
assert np.array_equal(output.asnumpy(), expected)
@pytest.mark.level0
@pytest.mark.platform_x86_gpu_training
@pytest.mark.env_onecard
def test_net_int32():
context.set_context(mode=context.PYNATIVE_MODE, device_target="GPU")
x = np.array([[1, 2, 3], [4, 5, 6], [7, 8, 9]]).astype(np.int32)
seq_lengths = np.array([1, 2, 3]).astype(np.int64)
seq_dim = 1
batch_dim = 0
net = Net(seq_dim, batch_dim)
output = net(Tensor(x), Tensor(seq_lengths))
expected = np.array([[1, 2, 3], [5, 4, 6], [9, 8, 7]]).astype(np.int32)
assert np.array_equal(output.asnumpy(), expected)
@pytest.mark.level0
@pytest.mark.platform_x86_gpu_training
@pytest.mark.env_onecard
def test_net_float32():
context.set_context(mode=context.GRAPH_MODE, device_target="GPU")
x = np.array([[[[1, 2], [3, 4]],
[[5, 6], [7, 8]],
[[9, 10], [11, 12]],
[[13, 14], [15, 16]]],
[[[17, 18], [19, 20]],
[[21, 22], [23, 24]],
[[25, 26], [27, 28]],
[[29, 30], [31, 21]]]]).astype(np.float32)
seq_lengths = np.array([2, 2, 2, 2]).astype(np.int64)
seq_dim = 0
batch_dim = 1
net = Net(seq_dim, batch_dim)
output = net(Tensor(x), Tensor(seq_lengths))
expected = np.array([[[[17., 18.], [19., 20.]],
[[21., 22.], [23., 24.]],
[[25., 26.], [27., 28.]],
[[29., 30.], [31., 21.]]],
[[[1., 2.], [3., 4.]],
[[5., 6.], [7., 8.]],
[[9., 10.], [11., 12.]],
[[13., 14.], [15., 16.]]]]).astype(np.float32)
assert np.array_equal(output.asnumpy(), expected)
@pytest.mark.level0
@pytest.mark.platform_x86_gpu_training
@pytest.mark.env_onecard
def test_net_float64_0_dim():
"""
Test added to test for 0 seq len edge case
"""
context.set_context(mode=context.GRAPH_MODE, device_target="GPU")
x = np.array([[[[1, 2], [3, 4]],
[[5, 6], [7, 8]],
[[9, 10], [11, 12]],
[[13, 14], [15, 16]]],
[[[17, 18], [19, 20]],
[[21, 22], [23, 24]],
[[25, 26], [27, 28]],
[[29, 30], [31, 21]]]]).astype(np.float32)
seq_lengths = np.array([2, 2, 0, 0]).astype(np.int64)
seq_dim = 2
batch_dim = 1
net = Net(seq_dim, batch_dim)
output = net(Tensor(x), Tensor(seq_lengths))
expected = np.array([[[[3., 4.], [1., 2.]],
[[7., 8.], [5., 6.]],
[[9., 10.], [11., 12.]],
[[13., 14.], [15., 16.]]],
[[[19., 20.], [17., 18.]],
[[23., 24.], [21., 22.]],
[[25., 26.], [27., 28.]],
[[29., 30.], [31., 21.]]]]).astype(np.float32)
assert np.array_equal(output.asnumpy(), expected)