gru_kernel.h 11.2 KB
Newer Older
1
/* Copyright (c) 2016 PaddlePaddle Authors. All Rights Reserved.
G
guosheng 已提交
2 3 4 5 6 7 8 9 10 11 12 13 14

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. */

15 16
#pragma once
#include <type_traits>
Y
Yi Wang 已提交
17 18
#include "paddle/fluid/operators/math/detail/activation_functions.h"
#include "paddle/fluid/platform/hostdevice.h"
G
guosheng 已提交
19

G
guosheng 已提交
20
// TODO(guosheng): refine code style in gru_kernel
G
guosheng 已提交
21 22 23 24 25 26 27 28 29 30
namespace paddle {
namespace operators {
namespace math {
namespace detail {

namespace forward {

template <typename T>
class gru_resetOutput {
 public:
31 32
  HOSTDEVICE void operator()(T *value_update_gate, T *value_reset_gate,
                             T *prev_out, T *value_reset_output,
33 34 35
                             ActivationType act_gate,
                             T *value_reset_bias = nullptr,
                             bool old_version = true) {
36 37
    *value_update_gate = activation(*value_update_gate, act_gate);
    *value_reset_gate = activation(*value_reset_gate, act_gate);
38 39 40 41 42 43
    if (old_version) {
      *value_reset_output = (*prev_out) * (*value_reset_gate);
    } else {
      *value_reset_output =
          (*value_reset_output + *value_reset_bias) * (*value_reset_gate);
    }
G
guosheng 已提交
44
  }
45
#if !defined(__NVCC__) && !defined(__HIPCC___)  // @{ Group GRU reset output
G
guosheng 已提交
46 47 48 49
#ifndef __AVX__
  static const bool avx = false;
#else
  static const bool avx = true;
50 51 52
  HOSTDEVICE void operator()(__m256 *value_update_gate,
                             __m256 *value_reset_gate, __m256 *prev_out,
                             __m256 *value_reset_output,
53 54 55
                             ActivationType act_gate,
                             __m256 *value_reset_bias = nullptr,
                             bool old_version = true) {
56 57
    *value_update_gate = activation(*value_update_gate, act_gate);
    *value_reset_gate = activation(*value_reset_gate, act_gate);
58 59 60 61 62 63 64 65
    if (old_version) {
      *value_reset_output = _mm256_mul_ps(*prev_out, *value_reset_gate);
    } else {
      *value_reset_output =
          _mm256_add_ps(*value_reset_output, *value_reset_bias);
      *value_reset_output =
          _mm256_mul_ps(*value_reset_output, *value_reset_gate);
    }
G
guosheng 已提交
66 67
  }
#endif
68
#endif  // @} End Group GRU reset output
G
guosheng 已提交
69 70 71 72 73
};

template <typename T>
class gru_finalOutput {
 public:
74 75
  HOSTDEVICE void operator()(T *value_update_gate, T *value_frame_state,
                             T *prev_out, T *value_output,
Q
Qiao Longfei 已提交
76
                             ActivationType act_input, bool origin_mode) {
77
    *value_frame_state = activation(*value_frame_state, act_input);
Q
Qiao Longfei 已提交
78 79 80 81 82 83 84 85
    if (origin_mode) {
      *value_output = ((*value_update_gate) * (*prev_out)) +
                      *value_frame_state -
                      ((*value_update_gate) * (*value_frame_state));
    } else {
      *value_output = *prev_out - ((*value_update_gate) * (*prev_out)) +
                      ((*value_update_gate) * (*value_frame_state));
    }
G
guosheng 已提交
86
  }
87
#if !defined(__NVCC__) && !defined(__HIPCC___)  // @{ Group GRU final output
G
guosheng 已提交
88 89 90 91
#ifndef __AVX__
  static const bool avx = false;
#else
  static const bool avx = true;
92 93
  HOSTDEVICE void operator()(__m256 *value_update_gate,
                             __m256 *value_frame_state, __m256 *prev_out,
Q
Qiao Longfei 已提交
94 95
                             __m256 *value_output, ActivationType act_input,
                             bool origin_mode) {
96
    *value_frame_state = activation(*value_frame_state, act_input);
Q
Qiao Longfei 已提交
97 98 99 100 101 102 103 104 105 106 107
    if (origin_mode) {
      *value_output = _mm256_sub_ps(
          _mm256_add_ps(_mm256_mul_ps(*value_update_gate, *prev_out),
                        *value_frame_state),
          _mm256_mul_ps(*value_update_gate, *value_frame_state));
    } else {
      *value_output = _mm256_add_ps(
          _mm256_sub_ps(*prev_out,
                        _mm256_mul_ps(*value_update_gate, *prev_out)),
          _mm256_mul_ps(*value_update_gate, *value_frame_state));
    }
G
guosheng 已提交
108 109
  }
#endif
110
#endif  // @} End Group GRU final output
G
guosheng 已提交
111 112 113 114 115 116 117 118
};
}  // namespace forward

namespace backward {

template <typename T>
class gru_stateGrad {
 public:
119 120 121
  HOSTDEVICE void operator()(T *value_update_gate, T *grad_update_gate,
                             T *value_frame_state, T *grad_frame_state,
                             T *value_prev_out, T *grad_prev_out,
Q
Qiao Longfei 已提交
122 123 124 125 126 127 128 129 130 131 132 133 134 135 136 137 138
                             T *grad_output, ActivationType act_input,
                             bool origin_mode) {
    if (origin_mode) {
      *grad_update_gate =
          (*grad_output) * ((*value_prev_out) - (*value_frame_state));
      *grad_prev_out += (*grad_output * (*value_update_gate));
      *grad_frame_state = activation(
          *grad_output * (static_cast<T>(1.0) - (*value_update_gate)),
          *value_frame_state, act_input);
    } else {
      *grad_update_gate =
          (*grad_output) * ((*value_frame_state) - (*value_prev_out));
      *grad_prev_out +=
          (*grad_output * (static_cast<T>(1.0) - *value_update_gate));
      *grad_frame_state = activation(*grad_output * (*value_update_gate),
                                     *value_frame_state, act_input);
    }
G
guosheng 已提交
139
  }
140
#if !defined(__NVCC__) && !defined(__HIPCC___)  // @{ Group GRU state grad
G
guosheng 已提交
141 142 143 144
#ifndef __AVX__
  static const bool avx = false;
#else
  static const bool avx = true;
145 146 147 148 149
  HOSTDEVICE void operator()(__m256 *value_update_gate,
                             __m256 *grad_update_gate,
                             __m256 *value_frame_state,
                             __m256 *grad_frame_state, __m256 *value_prev_out,
                             __m256 *grad_prev_out, __m256 *grad_output,
Q
Qiao Longfei 已提交
150
                             ActivationType act_input, bool origin_mode) {
151 152 153 154 155 156 157 158 159 160 161 162 163 164 165 166 167 168 169 170
    if (origin_mode) {
      *grad_update_gate = _mm256_mul_ps(
          *grad_output, _mm256_sub_ps(*value_prev_out, *value_frame_state));
      *grad_prev_out = _mm256_add_ps(
          *grad_prev_out, _mm256_mul_ps(*grad_output, *value_update_gate));
      *grad_frame_state = activation(
          _mm256_mul_ps(*grad_output, _mm256_sub_ps(_mm256_set1_ps(1.0f),
                                                    *value_update_gate)),
          *value_frame_state, act_input);
    } else {
      *grad_update_gate = _mm256_mul_ps(
          *grad_output, _mm256_sub_ps(*value_frame_state, *value_prev_out));
      *grad_prev_out = _mm256_add_ps(
          *grad_prev_out,
          _mm256_mul_ps(*grad_output, _mm256_sub_ps(_mm256_set1_ps(1.0f),
                                                    *value_update_gate)));
      *grad_frame_state =
          activation(_mm256_mul_ps(*grad_output, *value_update_gate),
                     *value_frame_state, act_input);
    }
G
guosheng 已提交
171 172
  }
#endif
173
#endif  // @} End Group GRU state grad
G
guosheng 已提交
174 175 176 177 178
};

template <typename T>
class gru_resetGrad {
 public:
179 180 181
  HOSTDEVICE void operator()(T *value_update_gate, T *grad_update_gate,
                             T *value_reset_gate, T *grad_reset_gate,
                             T *value_prev_out, T *grad_prev_out,
Q
Qiao Longfei 已提交
182
                             T *grad_reset_output, ActivationType act_gate) {
183 184 185 186 187 188
    *grad_reset_gate = (*grad_reset_output * (*value_prev_out));
    *grad_prev_out += (*grad_reset_output * (*value_reset_gate));
    *grad_update_gate =
        activation(*grad_update_gate, *value_update_gate, act_gate);
    *grad_reset_gate =
        activation(*grad_reset_gate, *value_reset_gate, act_gate);
G
guosheng 已提交
189
  }
190
#if !defined(__NVCC__) && !defined(__HIPCC___)  // @{ Group GRU reset grad
G
guosheng 已提交
191 192 193 194
#ifndef __AVX__
  static const bool avx = false;
#else
  static const bool avx = true;
195 196 197 198
  HOSTDEVICE void operator()(__m256 *value_update_gate,
                             __m256 *grad_update_gate, __m256 *value_reset_gate,
                             __m256 *grad_reset_gate, __m256 *value_prev_out,
                             __m256 *grad_prev_out, __m256 *grad_reset_output,
Q
Qiao Longfei 已提交
199
                             ActivationType act_gate) {
200 201 202 203 204 205 206
    *grad_reset_gate = _mm256_mul_ps(*grad_reset_output, *value_prev_out);
    *grad_prev_out = _mm256_add_ps(
        *grad_prev_out, _mm256_mul_ps(*grad_reset_output, *value_reset_gate));
    *grad_update_gate =
        activation(*grad_update_gate, *value_update_gate, act_gate);
    *grad_reset_gate =
        activation(*grad_reset_gate, *value_reset_gate, act_gate);
G
guosheng 已提交
207 208
  }
#endif
209
#endif  // @} End Group GRU reset grad
G
guosheng 已提交
210
};
211 212 213 214 215 216 217 218 219 220 221 222 223 224 225 226 227 228 229 230 231 232
template <typename T>
class gru {
 public:
  HOSTDEVICE void operator()(T *value_reset_gate, T *grad_reset_gate,
                             T *value_update_gate, T *grad_update_gate,
                             T *value_frame_state, T *grad_frame_state,
                             T *value_prev_out, T *grad_prev_out,
                             T *grad_output, T *value_reset_output,
                             T *grad_reset_output, ActivationType act_node,
                             ActivationType act_gate) {
    *grad_update_gate =
        activation((*grad_output) * ((*value_prev_out) - (*value_frame_state)),
                   (*value_update_gate), act_gate);
    *grad_prev_out += (*grad_output * (*value_update_gate));
    *grad_frame_state =
        activation(*grad_output * (static_cast<T>(1.0) - (*value_update_gate)),
                   *value_frame_state, act_node);
    T reset_output = (*value_reset_output) / (*value_reset_gate);
    *grad_reset_gate = activation(reset_output * (*grad_frame_state),
                                  *value_reset_gate, act_gate);
    *grad_reset_output = (*value_reset_gate) * (*grad_frame_state);
  }
233
#if !defined(__NVCC__) && !defined(__HIPCC___)  // @{ Group GRU CPU
234 235 236 237 238 239 240 241 242 243 244 245 246 247 248 249 250 251 252 253 254 255 256 257 258 259 260 261 262 263
#ifndef __AVX__
  static const bool avx = false;
#else
  static const bool avx = true;
  HOSTDEVICE void operator()(__m256 *value_reset_gate, __m256 *grad_reset_gate,
                             __m256 *value_update_gate,
                             __m256 *grad_update_gate,
                             __m256 *value_frame_state,
                             __m256 *grad_frame_state, __m256 *value_prev_out,
                             __m256 *grad_prev_out, __m256 *grad_output,
                             __m256 *value_reset_output,
                             __m256 *grad_reset_output, ActivationType act_node,
                             ActivationType act_gate) {
    *grad_update_gate = activation(
        _mm256_mul_ps(*grad_output,
                      _mm256_sub_ps(*value_prev_out, *value_frame_state)),
        *value_update_gate, act_gate);
    *grad_prev_out = _mm256_add_ps(
        *grad_prev_out, _mm256_mul_ps(*grad_output, *value_update_gate));
    *grad_frame_state = activation(
        _mm256_mul_ps(*grad_output,
                      _mm256_sub_ps(_mm256_set1_ps(1.0f), *value_update_gate)),
        *value_frame_state, act_node);
    __m256 reset_output = _mm256_div_ps(*value_reset_output, *value_reset_gate);
    *grad_reset_gate =
        activation(_mm256_mul_ps(reset_output, *grad_frame_state),
                   *value_reset_gate, act_gate);
    *grad_reset_output = _mm256_mul_ps(*value_reset_gate, *grad_frame_state);
  }
#endif
264
#endif  // @} End Group GRU CPU
265
};
G
guosheng 已提交
266 267 268 269 270 271 272

}  // namespace backward

}  // namespace detail
}  // namespace math
}  // namespace operators
}  // namespace paddle