pruner.py 7.6 KB
Newer Older
W
wanghaoshuang 已提交
1 2 3 4 5 6 7 8 9 10 11 12 13 14
# Copyright (c) 2019  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.

15
import logging
16
import sys
W
wanghaoshuang 已提交
17
import numpy as np
18
from functools import reduce
W
wanghaoshuang 已提交
19
import paddle.fluid as fluid
20 21
import copy
from ..core import VarWrapper, OpWrapper, GraphWrapper
22 23 24
from .group_param import collect_convs
from .criterion import CRITERION
from .idx_selector import IDX_SELECTOR
25
from ..common import get_logger
W
wanghaoshuang 已提交
26

27
__all__ = ["Pruner"]
W
wanghaoshuang 已提交
28

29 30
_logger = get_logger(__name__, level=logging.INFO)

W
wanghaoshuang 已提交
31 32

class Pruner():
33 34 35
    """The pruner used to prune channels of convolution.

    Args:
36 37
        criterion(str|function): the criterion used to sort channels for pruning.
        idx_selector(str|function): 
38 39 40

    """

41 42 43 44 45 46 47 48 49 50 51 52 53
    def __init__(self,
                 criterion="l1_norm",
                 idx_selector="default_idx_selector"):
        if isinstance(criterion, str):
            self.criterion = CRITERION.get(criterion)
        else:
            self.criterion = criterion
        if isinstance(idx_selector, str):
            self.idx_selector = IDX_SELECTOR.get(idx_selector)
        else:
            self.idx_selector = idx_selector

        self.pruned_weights = False
W
wanghaoshuang 已提交
54 55 56 57 58 59 60 61 62

    def prune(self,
              program,
              scope,
              params,
              ratios,
              place=None,
              lazy=False,
              only_graph=False,
W
wanghaoshuang 已提交
63 64
              param_backup=False,
              param_shape_backup=False):
65 66
        """Pruning the given parameters.

W
wanghaoshuang 已提交
67
        Args:
68

W
wanghaoshuang 已提交
69 70 71 72 73 74 75 76 77
            program(fluid.Program): The program to be pruned.
            scope(fluid.Scope): The scope storing paramaters to be pruned.
            params(list<str>): A list of parameter names to be pruned.
            ratios(list<float>): A list of ratios to be used to pruning parameters.
            place(fluid.Place): The device place of filter parameters. Defalut: None.
            lazy(bool): True means setting the pruned elements to zero.
                        False means cutting down the pruned elements. Default: False.
            only_graph(bool): True means only modifying the graph.
                              False means modifying graph and variables in scope. Default: False.
W
wanghaoshuang 已提交
78 79
            param_backup(bool): Whether to return a dict to backup the values of parameters. Default: False.
            param_shape_backup(bool): Whether to return a dict to backup the shapes of parameters. Default: False.
80

W
wanghaoshuang 已提交
81
        Returns:
82
            tuple: ``(pruned_program, param_backup, param_shape_backup)``. ``pruned_program`` is the pruned program. ``param_backup`` is a dict to backup the values of parameters. ``param_shape_backup`` is a dict to backup the shapes of parameters.
W
wanghaoshuang 已提交
83 84 85 86
        """

        self.pruned_list = []
        graph = GraphWrapper(program.clone())
W
wanghaoshuang 已提交
87 88
        param_backup = {} if param_backup else None
        param_shape_backup = {} if param_shape_backup else None
W
wanghaoshuang 已提交
89

W
wanghaoshuang 已提交
90
        visited = {}
W
whs 已提交
91
        pruned_params = []
W
wanghaoshuang 已提交
92
        for param, ratio in zip(params, ratios):
93
            group = collect_convs([param], graph, visited)[0]  # [(name, axis)]
W
whs 已提交
94 95
            if group is None or len(group) == 0:
                continue
96 97
            if only_graph and self.idx_selector.__name__ == "default_idx_selector":

W
whs 已提交
98 99
                param_v = graph.var(param)
                pruned_num = int(round(param_v.shape()[0] * ratio))
100
                pruned_idx = [0] * pruned_num
101
                for name, axis in group:
102 103
                    pruned_params.append((name, axis, pruned_idx))

W
whs 已提交
104
            else:
105 106 107 108 109 110 111
                assert ((not self.pruned_weights),
                        "The weights have been pruned once.")
                group_values = []
                for name, axis in group:
                    values = np.array(scope.find_var(name).get_tensor())
                    group_values.append((name, values, axis))

112
                scores = self.criterion(group_values,
113 114
                                        graph)  # [(name, axis, score)]

115
                pruned_params.extend(self.idx_selector(scores, ratio))
W
whs 已提交
116 117 118

        merge_pruned_params = {}
        for param, pruned_axis, pruned_idx in pruned_params:
119 120 121 122 123
            if param not in merge_pruned_params:
                merge_pruned_params[param] = {}
            if pruned_axis not in merge_pruned_params[param]:
                merge_pruned_params[param][pruned_axis] = []
            merge_pruned_params[param][pruned_axis].append(pruned_idx)
W
whs 已提交
124 125 126

        for param_name in merge_pruned_params:
            for pruned_axis in merge_pruned_params[param_name]:
W
whs 已提交
127 128
                pruned_idx = np.concatenate(merge_pruned_params[param_name][
                    pruned_axis])
W
whs 已提交
129
                param = graph.var(param_name)
W
whs 已提交
130
                if not lazy:
131 132 133
                    _logger.debug("{}\t{}\t{}\t{}".format(
                        param.name(), pruned_axis,
                        param.shape()[pruned_axis], len(pruned_idx)))
W
whs 已提交
134 135 136 137 138 139
                    if param_shape_backup is not None:
                        origin_shape = copy.deepcopy(param.shape())
                        param_shape_backup[param.name()] = origin_shape
                    new_shape = list(param.shape())
                    new_shape[pruned_axis] -= len(pruned_idx)
                    param.set_shape(new_shape)
W
whs 已提交
140 141
                if not only_graph:
                    param_t = scope.find_var(param.name()).get_tensor()
W
whs 已提交
142 143 144 145
                    if param_backup is not None and (
                            param.name() not in param_backup):
                        param_backup[param.name()] = copy.deepcopy(
                            np.array(param_t))
W
whs 已提交
146 147 148 149 150 151 152
                    try:
                        pruned_param = self._prune_tensor(
                            np.array(param_t),
                            pruned_idx,
                            pruned_axis=pruned_axis,
                            lazy=lazy)
                    except IndexError as e:
W
whs 已提交
153 154 155
                        _logger.error("Pruning {}, but get [{}]".format(
                            param.name(), e))

W
whs 已提交
156
                    param_t.set(pruned_param, place)
W
whs 已提交
157
        graph.update_groups_of_conv()
158
        graph.infer_shape()
159
        self.pruned_weights = (not only_graph)
W
whs 已提交
160
        return graph.program, param_backup, param_shape_backup
W
wanghaoshuang 已提交
161 162 163 164

    def _prune_tensor(self, tensor, pruned_idx, pruned_axis, lazy=False):
        """
        Pruning a array by indexes on given axis.
165

W
wanghaoshuang 已提交
166 167 168 169 170 171 172
        Args:
            tensor(numpy.array): The target array to be pruned.
            pruned_idx(list<int>): The indexes to be pruned.
            pruned_axis(int): The axis of given array to be pruned on. 
            lazy(bool): True means setting the pruned elements to zero.
                        False means remove the pruned elements from memory.
                        default: False.
173

W
wanghaoshuang 已提交
174 175 176 177 178 179 180 181 182 183 184 185 186 187 188 189 190
        Returns:
            numpy.array: The pruned array.
        """
        mask = np.zeros(tensor.shape[pruned_axis], dtype=bool)
        mask[pruned_idx] = True

        def func(data):
            return data[~mask]

        def lazy_func(data):
            data[mask] = 0
            return data

        if lazy:
            return np.apply_along_axis(lazy_func, pruned_axis, tensor)
        else:
            return np.apply_along_axis(func, pruned_axis, tensor)