elementwise_add_1d.cpp 3.94 KB
Newer Older
Chao Liu's avatar
Chao Liu committed
1
2
3
// SPDX-License-Identifier: MIT
// Copyright (c) 2018-2022, Advanced Micro Devices, Inc. All rights reserved.

rocking5566's avatar
rocking5566 committed
4
5
6
#include <iostream>
#include <cstdlib>

Chao Liu's avatar
Chao Liu committed
7
#include "ck/ck.hpp"
8
#include "ck/tensor_operation/gpu/device/impl/device_elementwise.hpp"
Chao Liu's avatar
Chao Liu committed
9
10
#include "ck/tensor_operation/gpu/element/binary_element_wise_operation.hpp"
#include "ck/library/utility/check_err.hpp"
11
12
13
#include "ck/library/utility/device_memory.hpp"
#include "ck/library/utility/host_tensor.hpp"
#include "ck/library/utility/host_tensor_generator.hpp"
rocking5566's avatar
rocking5566 committed
14
15
16
17

using F16 = ck::half_t;
using F32 = float;

18
19
using ABDataType = F16;
using CDataType  = F16;
rocking5566's avatar
rocking5566 committed
20

21
using Add = ck::tensor_operation::element_wise::Add;
rocking5566's avatar
rocking5566 committed
22

23
using DeviceElementwiseAddInstance =
24
25
26
27
28
29
30
31
32
    ck::tensor_operation::device::DeviceElementwise<ck::Tuple<ABDataType, ABDataType>,
                                                    ck::Tuple<CDataType>,
                                                    Add,
                                                    1,
                                                    8,
                                                    ck::Sequence<8, 8>,
                                                    ck::Sequence<8>>;

template <typename HostTensorA, typename HostTensorB, typename HostTensorC, typename Functor>
rocking5566's avatar
rocking5566 committed
33
34
35
36
37
38
39
void host_elementwise1D(
    HostTensorC& C, const HostTensorA& A, const HostTensorB& B, int M, Functor functor)
{
    using ctype = ck::remove_reference_t<decltype(C(0))>;

    for(int m = 0; m < M; ++m)
    {
40
41
42
        auto Am  = A(m);
        auto Bm  = B(m);
        ctype Cm = 0;
rocking5566's avatar
rocking5566 committed
43
        functor(Cm, Am, Bm);
44
        C(m) = Cm;
rocking5566's avatar
rocking5566 committed
45
46
47
48
49
50
51
52
53
54
55
    }
}

int main()
{
    bool do_verification = true;
    bool time_kernel     = false;

    ck::index_t M = 1024;

    auto f_host_tensor_descriptor1d = [](std::size_t len, std::size_t stride) {
56
        return HostTensorDescriptor({len}, {stride});
rocking5566's avatar
rocking5566 committed
57
58
59
60
    };

    Tensor<ABDataType> a_m(f_host_tensor_descriptor1d(M, 1));
    Tensor<ABDataType> b_m(f_host_tensor_descriptor1d(M, 1));
rocking5566's avatar
rocking5566 committed
61
    Tensor<CDataType> c_m(f_host_tensor_descriptor1d(M, 1));
rocking5566's avatar
rocking5566 committed
62
63
64
65

    a_m.GenerateTensorValue(GeneratorTensor_3<ABDataType>{0.0, 1.0});
    b_m.GenerateTensorValue(GeneratorTensor_3<ABDataType>{0.0, 1.0});

66
67
68
    DeviceMem a_m_device_buf(sizeof(ABDataType) * a_m.mDesc.GetElementSpaceSize());
    DeviceMem b_m_device_buf(sizeof(ABDataType) * b_m.mDesc.GetElementSpaceSize());
    DeviceMem c_m_device_buf(sizeof(CDataType) * c_m.mDesc.GetElementSpaceSize());
rocking5566's avatar
rocking5566 committed
69
70
71
72

    a_m_device_buf.ToDevice(a_m.mData.data());
    b_m_device_buf.ToDevice(b_m.mData.data());

73
74
75
76
    std::array<const void*, 2> input = {a_m_device_buf.GetDeviceBuffer(),
                                        b_m_device_buf.GetDeviceBuffer()};
    std::array<void*, 1> output      = {c_m_device_buf.GetDeviceBuffer()};

77
78
79
80
    std::array<ck::index_t, 1> abc_lengths = {M};
    std::array<ck::index_t, 1> a_strides   = {1};
    std::array<ck::index_t, 1> b_strides   = {1};
    std::array<ck::index_t, 1> c_strides   = {1};
81

rocking5566's avatar
rocking5566 committed
82
    auto broadcastAdd = DeviceElementwiseAddInstance{};
83
    auto argument     = broadcastAdd.MakeArgumentPointer(
84
        abc_lengths, {a_strides, b_strides}, {c_strides}, input, output, Add{});
rocking5566's avatar
rocking5566 committed
85
86
87

    if(!broadcastAdd.IsSupportedArgument(argument.get()))
    {
88
89
        throw std::runtime_error(
            "The runtime parameters seems not supported by the device instance, exiting!");
rocking5566's avatar
rocking5566 committed
90
91
92
93
94
95
96
97
98
99
100
101
102
103
    };

    auto broadcastAdd_invoker_ptr = broadcastAdd.MakeInvokerPointer();
    float ave_time =
        broadcastAdd_invoker_ptr->Run(argument.get(), StreamConfig{nullptr, time_kernel});

    std::cout << "Perf: " << ave_time << " ms" << std::endl;

    bool pass = true;
    if(do_verification)
    {
        c_m_device_buf.FromDevice(c_m.mData.data());
        Tensor<CDataType> host_c_m(f_host_tensor_descriptor1d(M, 1));

104
105
        host_elementwise1D<Tensor<ABDataType>, Tensor<ABDataType>, Tensor<CDataType>, Add>(
            host_c_m, a_m, b_m, M, Add{});
rocking5566's avatar
rocking5566 committed
106

107
        pass &= ck::utils::check_err(c_m, host_c_m, "Error: Incorrect results c", 1e-3, 1e-3);
rocking5566's avatar
rocking5566 committed
108
109
110
111
    }

    return pass ? 0 : 1;
}