pruner.py 9.5 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 18
import numpy as np
import paddle.fluid as fluid
19 20
import copy
from ..core import VarWrapper, OpWrapper, GraphWrapper
21 22 23
from .group_param import collect_convs
from .criterion import l1_norm
from .importance_sort import channel_score_sort, batch_norm_scale
24
from ..common import get_logger
W
wanghaoshuang 已提交
25

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

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

W
wanghaoshuang 已提交
30 31

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

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

    """

40
    def __init__(self, criterion="l1_norm", channel_sortor="channel_score"):
W
wanghaoshuang 已提交
41
        self.criterion = criterion
42 43 44 45 46 47 48 49
        self.channel_sortor = channel_sortor
        if criterion == "l1_norm":
            self.criterion = l1_norm

        if channel_sortor == "channel_score":
            self.channel_sortor = channel_score_sort
        elif channel_sortor == "batch_norm_scale":
            self.channel_sortor = batch_norm_scale_sort
W
wanghaoshuang 已提交
50 51

    def prune(self,
W
wanghaoshuang 已提交
52
              graph,
W
wanghaoshuang 已提交
53 54 55 56 57 58
              scope,
              params,
              ratios,
              place=None,
              lazy=False,
              only_graph=False,
W
wanghaoshuang 已提交
59 60
              param_backup=False,
              param_shape_backup=False):
61 62
        """Pruning the given parameters.

W
wanghaoshuang 已提交
63
        Args:
64

W
wanghaoshuang 已提交
65 66 67 68 69 70 71 72 73
            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 已提交
74 75
            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.
76

W
wanghaoshuang 已提交
77
        Returns:
78
            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 已提交
79 80 81
        """

        self.pruned_list = []
W
wanghaoshuang 已提交
82 83 84 85 86 87 88
        if isinstance(graph, fluid.Program):
            graph = GraphWrapper(program.clone())
        elif isinstance(graph, torch.nn.Module):
            graph = DyGraph(graph)
            conv2d_walker = dy_conv2d_walker
        else:
            raise NotImplementedError('The type of graph is not supported.')
W
wanghaoshuang 已提交
89 90
        param_backup = {} if param_backup else None
        param_shape_backup = {} if param_shape_backup else None
W
wanghaoshuang 已提交
91

W
wanghaoshuang 已提交
92
        visited = {}
W
whs 已提交
93
        pruned_params = []
W
wanghaoshuang 已提交
94
        for param, ratio in zip(params, ratios):
95
            group = collect_convs([param], graph)[0]  # [(name, axis)]
W
whs 已提交
96
            if only_graph:
97

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

W
whs 已提交
104
            else:
105 106 107 108 109 110 111 112 113 114 115 116 117 118 119 120

                group_values = []
                for name, axis in group:
                    values = np.array(scope.find_var(name).get_tensor())
                    group_values.append((name, values, axis))

                scores = self.criterion(
                    group_with_values)  # [(name, axis, score)]

                group_idx = self.channel_sortor(
                    scores, graph=graph)  # [(name, axis, soted_idx)]
                for param, pruned_axis, pruned_idx in group_idx:
                    pruned_num = len(pruned_idx) * ratio
                    pruned_params.append((
                        param, pruned_axis,
                        pruned_idx[:pruned_num]))  # [(name, axis, pruned_idx)]
W
whs 已提交
121 122 123

        merge_pruned_params = {}
        for param, pruned_axis, pruned_idx in pruned_params:
124 125 126 127 128
            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 已提交
129 130 131

        for param_name in merge_pruned_params:
            for pruned_axis in merge_pruned_params[param_name]:
W
whs 已提交
132 133
                pruned_idx = np.concatenate(merge_pruned_params[param_name][
                    pruned_axis])
W
whs 已提交
134
                param = graph.var(param_name)
W
whs 已提交
135 136 137 138 139 140 141 142 143
                if not lazy:
                    _logger.debug("{}\t{}\t{}".format(param.name(
                    ), pruned_axis, len(pruned_idx)))
                    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 已提交
144 145
                if not only_graph:
                    param_t = scope.find_var(param.name()).get_tensor()
W
whs 已提交
146 147 148 149
                    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 已提交
150 151 152 153 154 155 156
                    try:
                        pruned_param = self._prune_tensor(
                            np.array(param_t),
                            pruned_idx,
                            pruned_axis=pruned_axis,
                            lazy=lazy)
                    except IndexError as e:
W
whs 已提交
157 158 159
                        _logger.error("Pruning {}, but get [{}]".format(
                            param.name(), e))

W
whs 已提交
160
                    param_t.set(pruned_param, place)
W
whs 已提交
161
        graph.update_groups_of_conv()
162
        graph.infer_shape()
W
whs 已提交
163
        return graph.program, param_backup, param_shape_backup
W
wanghaoshuang 已提交
164

165
    def _cal_pruned_idx(self, graph, scope, param, ratio, axis):
W
wanghaoshuang 已提交
166 167
        """
        Calculate the index to be pruned on axis by given pruning ratio.
168

W
wanghaoshuang 已提交
169 170 171 172 173 174 175
        Args:
            name(str): The name of parameter to be pruned.
            param(np.array): The data of parameter to be pruned.
            ratio(float): The ratio to be pruned.
            axis(int): The axis to be used for pruning given parameter.
                       If it is None, the value in self.pruning_axis will be used.
                       default: None.
176

W
wanghaoshuang 已提交
177 178 179 180
        Returns:
            list<int>: The indexes to be pruned on axis.
        """
        if self.criterion == 'l1_norm':
181 182 183 184 185 186 187 188 189 190 191 192 193 194 195 196 197 198 199
            param_t = np.array(scope.find_var(param).get_tensor())
            prune_num = int(round(param_t.shape[axis] * ratio))
            reduce_dims = [i for i in range(len(param_t.shape)) if i != axis]
            criterions = np.sum(np.abs(param_t), axis=tuple(reduce_dims))
            pruned_idx = criterions.argsort()[:prune_num]
        elif self.criterion == "batch_norm_scale":
            param_var = graph.var(param)
            conv_op = param_var.outputs()[0]
            conv_output = conv_op.outputs("Output")[0]
            bn_op = conv_output.outputs()[0]
            if bn_op is not None:
                bn_scale_param = bn_op.inputs("Scale")[0].name()
                bn_scale_np = np.array(
                    scope.find_var(bn_scale_param).get_tensor())
                prune_num = int(round(bn_scale_np.shape[axis] * ratio))
                pruned_idx = np.abs(bn_scale_np).argsort()[:prune_num]
            else:
                raise SystemExit(
                    "Can't find BatchNorm op after Conv op in Network.")
W
wanghaoshuang 已提交
200 201 202 203 204
        return pruned_idx

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

W
wanghaoshuang 已提交
206 207 208 209 210 211 212
        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.
213

W
wanghaoshuang 已提交
214 215 216 217 218 219 220 221 222 223 224 225 226 227 228 229 230
        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)