File: layer_normalization_layer.hpp

package info (click to toggle)
frugally-deep 0.18.2-1
  • links: PTS, VCS
  • area: main
  • in suites: forky, sid
  • size: 1,036 kB
  • sloc: cpp: 6,680; python: 1,262; makefile: 4; sh: 1
file content (67 lines) | stat: -rw-r--r-- 2,311 bytes parent folder | download | duplicates (2)
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
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
// Copyright 2016, Tobias Hermann.
// https://github.com/Dobiasd/frugally-deep
// Distributed under the MIT License.
// (See accompanying LICENSE file or at
//  https://opensource.org/licenses/MIT)

#pragma once

#include "fdeep/layers/layer.hpp"

#include <string>

namespace fdeep {
namespace internal {

    class layer_normalization_layer : public layer {
    public:
        explicit layer_normalization_layer(const std::string& name,
            std::vector<int> axes,
            const float_vec& beta,
            const float_vec& gamma,
            float_type epsilon)
            : layer(name)
            , axes_(axes)
            , beta_(fplus::make_shared_ref<float_vec>(beta))
            , gamma_(fplus::make_shared_ref<float_vec>(gamma))
            , epsilon_(epsilon)
        {
        }

    protected:
        std::vector<int> axes_;
        shared_float_vec beta_;
        shared_float_vec gamma_;
        float_type epsilon_;

        tensors apply_impl(const tensors& inputs) const override
        {
            const auto& input = single_tensor_from_tensors(inputs);

            // https://github.com/keras-team/keras/blob/v2.14.0/keras/layers/normalization/layer_normalization.py#L291-L304
            const auto& input_moments = moments(input, axes_);
            const auto& mean = input_moments.first;
            const auto& variance = input_moments.second;

            std::vector<std::size_t> dims(5, 1);
            tensor_shape input_shape = input.shape();
            input_shape.maximize_rank();
            const auto input_shape_dimensions = input_shape.dimensions();
            for (const auto axis : axes_) {
                const std::size_t pos = rank_aligned_axis_to_absolute_axis(input.shape().rank(), axis) - 1;
                dims[pos] = input_shape_dimensions[pos];
            }
            const tensor_shape params_shape = create_tensor_shape_from_dims(dims);

            return { batch_normalization(
                input,
                mean,
                variance,
                beta_->empty() ? tensor(input.shape(), 0) : broadcast(tensor(params_shape, beta_), input.shape()),
                gamma_->empty() ? tensor(input.shape(), 1) : broadcast(tensor(params_shape, gamma_), input.shape()),
                epsilon_) };
        }
    };

}
}