pruner.py 8.3 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
W
whs 已提交
21
from .prune_walker import conv2d as conv2d_walker
22
from ..common import get_logger
W
wanghaoshuang 已提交
23

24
__all__ = ["Pruner"]
W
wanghaoshuang 已提交
25

26 27
_logger = get_logger(__name__, level=logging.INFO)

W
wanghaoshuang 已提交
28 29

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

    Args:
        criterion(str): the criterion used to sort channels for pruning. It only supports 'l1_norm' currently.

    """

W
wanghaoshuang 已提交
37 38 39 40 41 42 43 44 45 46 47
    def __init__(self, criterion="l1_norm"):
        self.criterion = criterion

    def prune(self,
              program,
              scope,
              params,
              ratios,
              place=None,
              lazy=False,
              only_graph=False,
W
wanghaoshuang 已提交
48 49
              param_backup=False,
              param_shape_backup=False):
50 51
        """Pruning the given parameters.

W
wanghaoshuang 已提交
52
        Args:
53

W
wanghaoshuang 已提交
54 55 56 57 58 59 60 61 62
            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 已提交
63 64
            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.
65

W
wanghaoshuang 已提交
66
        Returns:
67
            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 已提交
68 69 70 71
        """

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

W
wanghaoshuang 已提交
75
        visited = {}
W
whs 已提交
76
        pruned_params = []
W
wanghaoshuang 已提交
77
        for param, ratio in zip(params, ratios):
W
whs 已提交
78 79 80 81 82
            if only_graph:
                param_v = graph.var(param)
                pruned_num = int(round(param_v.shape()[0] * ratio))
                pruned_idx = [0] * pruned_num
            else:
83 84
                pruned_idx = self._cal_pruned_idx(
                    graph, scope, param, ratio, axis=0)
W
wanghaoshuang 已提交
85
            param = graph.var(param)
W
whs 已提交
86
            conv_op = param.outputs()[0]
W
whs 已提交
87 88
            walker = conv2d_walker(
                conv_op, pruned_params=pruned_params, visited=visited)
W
whs 已提交
89 90 91 92 93 94 95 96 97 98 99 100
            walker.prune(param, pruned_axis=0, pruned_idx=pruned_idx)

        merge_pruned_params = {}
        for param, pruned_axis, pruned_idx in pruned_params:
            if param.name() not in merge_pruned_params:
                merge_pruned_params[param.name()] = {}
            if pruned_axis not in merge_pruned_params[param.name()]:
                merge_pruned_params[param.name()][pruned_axis] = []
            merge_pruned_params[param.name()][pruned_axis].append(pruned_idx)

        for param_name in merge_pruned_params:
            for pruned_axis in merge_pruned_params[param_name]:
W
whs 已提交
101 102
                pruned_idx = np.concatenate(merge_pruned_params[param_name][
                    pruned_axis])
W
whs 已提交
103
                param = graph.var(param_name)
W
whs 已提交
104 105 106 107 108 109 110 111 112
                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 已提交
113 114
                if not only_graph:
                    param_t = scope.find_var(param.name()).get_tensor()
W
whs 已提交
115 116 117 118
                    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 已提交
119 120 121 122 123 124 125
                    try:
                        pruned_param = self._prune_tensor(
                            np.array(param_t),
                            pruned_idx,
                            pruned_axis=pruned_axis,
                            lazy=lazy)
                    except IndexError as e:
W
whs 已提交
126 127 128
                        _logger.error("Pruning {}, but get [{}]".format(
                            param.name(), e))

W
whs 已提交
129
                    param_t.set(pruned_param, place)
W
whs 已提交
130
        graph.update_groups_of_conv()
131
        graph.infer_shape()
W
whs 已提交
132
        return graph.program, param_backup, param_shape_backup
W
wanghaoshuang 已提交
133

134
    def _cal_pruned_idx(self, graph, scope, param, ratio, axis):
W
wanghaoshuang 已提交
135 136
        """
        Calculate the index to be pruned on axis by given pruning ratio.
137

W
wanghaoshuang 已提交
138 139 140 141 142 143 144
        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.
145

W
wanghaoshuang 已提交
146 147 148 149
        Returns:
            list<int>: The indexes to be pruned on axis.
        """
        if self.criterion == 'l1_norm':
150 151 152 153 154 155 156 157 158 159 160 161 162 163 164 165 166 167 168
            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 已提交
169 170 171 172 173
        return pruned_idx

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

W
wanghaoshuang 已提交
175 176 177 178 179 180 181
        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.
182

W
wanghaoshuang 已提交
183 184 185 186 187 188 189 190 191 192 193 194 195 196 197 198 199
        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)