topology.py 3.4 KB
Newer Older
Q
qiaolongfei 已提交
1 2 3 4 5 6 7 8 9 10 11 12 13 14
# Copyright (c) 2016 PaddlePaddle Authors. All Rights Reserved
#
# 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.

Q
qiaolongfei 已提交
15 16 17
import collections

from paddle.proto.ModelConfig_pb2 import ModelConfig
X
xuwei06 已提交
18
import paddle.trainer_config_helpers as conf_helps
Q
qiaolongfei 已提交
19
import layer as v2_layer
X
xuwei06 已提交
20
import config_base
21
import cPickle
Q
qiaolongfei 已提交
22 23 24 25 26 27 28 29 30 31

__all__ = ['Topology']


class Topology(object):
    """
    Topology is used to store the information about all layers
    and network configs.
    """

32 33 34 35 36 37 38 39 40
    def __init__(self, layers, extra_layers=None):
        def __check__(layers):
            if not isinstance(layers, collections.Sequence):
                layers = [layers]
            for layer in layers:
                __check_layer_type__(layer)
            return layers

        layers = __check__(layers)
Q
qiaolongfei 已提交
41
        self.layers = layers
42 43 44 45
        if extra_layers is not None:
            extra_layers = __check__(extra_layers)

        self.__model_config__ = v2_layer.parse_network(
D
dangqingqing 已提交
46
            layers, extra_layers=extra_layers)
D
bug fix  
dangqingqing 已提交
47 48 49 50

        if extra_layers is not None:
            self.layers.extend(extra_layers)

Q
qiaolongfei 已提交
51
        assert isinstance(self.__model_config__, ModelConfig)
Q
qiaolongfei 已提交
52

Q
qiaolongfei 已提交
53 54 55 56 57
    def use_sparse_updater(self):
        """
        check if any parameter require to use sparse_update
        :return:
        """
58
        use_sparse = False
Q
qiaolongfei 已提交
59 60
        for parameter in self.__model_config__.parameters:
            if parameter.sparse_update or parameter.sparse_remote_update:
61 62 63
                use_sparse = True
                break
        return use_sparse
Q
qiaolongfei 已提交
64

Q
qiaolongfei 已提交
65
    def proto(self):
Q
qiaolongfei 已提交
66 67 68
        return self.__model_config__

    def get_layer(self, name):
Q
qiaolongfei 已提交
69 70 71 72 73
        """
        get v2.Layer Class instance by layer name
        :param name:
        :return:
        """
X
xuwei06 已提交
74
        return v2_layer.get_layer(name)
Q
qiaolongfei 已提交
75

Q
qiaolongfei 已提交
76
    def data_layers(self):
Q
qiaolongfei 已提交
77 78 79 80
        """
        get all data layer
        :return:
        """
X
xuwei06 已提交
81 82 83 84 85
        data_layers = {}
        for layer in self.proto().layers:
            l = v2_layer.get_layer(layer.name)
            if l and l.layer_type == conf_helps.LayerType.DATA:
                data_layers[layer.name] = l
Q
qiaolongfei 已提交
86 87
        return data_layers

Q
qiaolongfei 已提交
88 89
    def data_type(self):
        """
Q
qiaolongfei 已提交
90 91
        get data_type from proto, such as:
        [('image', dense_vector(768)), ('label', integer_value(10))]
Q
qiaolongfei 已提交
92
        """
D
dangqingqing 已提交
93
        data_layers = self.data_layers()
94

X
xuwei06 已提交
95
        return [(nm, data_layers[nm].data_type)
C
caoying03 已提交
96
                for nm in self.proto().input_layer_names]
Q
qiaolongfei 已提交
97

Q
qiaolongfei 已提交
98 99 100 101 102 103
    def get_layer_proto(self, name):
        for layer in self.__model_config__.layers:
            if layer.name == name:
                return layer
        return None

104 105 106 107 108 109 110 111
    def serialize_for_inference(self, stream):
        protobin = self.proto().SerializeToString()
        data_type = self.data_type()
        cPickle.dump({
            'protobin': protobin,
            'data_type': data_type
        }, stream, cPickle.HIGHEST_PROTOCOL)

Q
qiaolongfei 已提交
112

Q
qiaolongfei 已提交
113
def __check_layer_type__(layer):
X
xuwei06 已提交
114 115
    if not isinstance(layer, config_base.Layer):
        raise ValueError('layer should have type paddle.v2.config_base.Layer')