convert.py 9.4 KB
Newer Older
J
jiangjiajun 已提交
1
# Copyright (c) 2020  PaddlePaddle Authors. All Rights Reserved.
J
jiangjiajun 已提交
2 3 4 5 6 7 8 9 10 11 12 13
#
# 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.
S
SunAhong1993 已提交
14

15 16
from six import text_type as _text_type
import argparse
J
jiangjiajun 已提交
17
import sys
18

J
jiangjiajun 已提交
19

20 21
def arg_parser():
    parser = argparse.ArgumentParser()
J
jiangjiajun 已提交
22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 41 42 43 44 45
    parser.add_argument(
        "--model",
        "-m",
        type=_text_type,
        default=None,
        help="define model file path for tensorflow or onnx")
    parser.add_argument(
        "--prototxt",
        "-p",
        type=_text_type,
        default=None,
        help="prototxt file of caffe model")
    parser.add_argument(
        "--weight",
        "-w",
        type=_text_type,
        default=None,
        help="weight file of caffe model")
    parser.add_argument(
        "--save_dir",
        "-s",
        type=_text_type,
        default=None,
        help="path to save translated model")
J
upgrade  
jiangjiajun 已提交
46 47 48 49 50
    parser.add_argument(
        "--framework",
        "-f",
        type=_text_type,
        default=None,
J
jiangjiajun 已提交
51 52 53
        help=
        "define which deeplearning framework(tensorflow/caffe/onnx/paddle2onnx)"
    )
S
SunAhong1993 已提交
54 55 56 57 58
    parser.add_argument(
        "--caffe_proto",
        "-c",
        type=_text_type,
        default=None,
J
upgrade  
jiangjiajun 已提交
59 60
        help="optional: the .py file compiled by caffe proto file of caffe model"
    )
J
jiangjiajun 已提交
61 62 63 64 65 66
    parser.add_argument(
        "--version",
        "-v",
        action="store_true",
        default=False,
        help="get version of x2paddle")
67 68 69 70 71 72
    parser.add_argument(
        "--without_data_format_optimization",
        "-wo",
        action="store_true",
        default=False,
        help="tf model conversion without data format optimization")
J
jiangjiajun 已提交
73 74 75 76 77 78 79 80 81 82 83 84
    parser.add_argument(
        "--define_input_shape",
        "-d",
        action="store_true",
        default=False,
        help="define input shape for tf model")
    parser.add_argument(
        "--params_merge",
        "-pm",
        action="store_true",
        default=False,
        help="define whether merge the params")
J
jiangjiajun 已提交
85

86
    return parser
J
jiangjiajun 已提交
87

88

89 90 91
def tf2paddle(model_path,
              save_dir,
              without_data_format_optimization=False,
M
mamingjie-China 已提交
92
              define_input_shape=False,
M
mamingjie-China 已提交
93
              params_merge=False):
J
jiangjiajun 已提交
94 95
    # check tensorflow installation and version
    try:
96 97
        import os
        os.environ["TF_CPP_MIN_LOG_LEVEL"] = '3'
J
jiangjiajun 已提交
98 99 100 101
        import tensorflow as tf
        version = tf.__version__
        if version >= '2.0.0' or version < '1.0.0':
            print(
J
jiangjiajun@baidu.com 已提交
102
                "[ERROR] 1.0.0<=tensorflow<2.0.0 is required, and v1.14.0 is recommended"
J
jiangjiajun 已提交
103 104 105
            )
            return
    except:
J
jiangjiajun@baidu.com 已提交
106 107 108
        print(
            "[ERROR] Tensorflow is not installed, use \"pip install tensorflow\"."
        )
J
jiangjiajun 已提交
109 110
        return

J
jiangjiajun 已提交
111
    from x2paddle.decoder.tf_decoder import TFDecoder
J
jiangjiajun 已提交
112
    from x2paddle.op_mapper.tf_op_mapper import TFOpMapper
113
    from x2paddle.op_mapper.tf_op_mapper_nhwc import TFOpMapperNHWC
J
jiangjiajun 已提交
114
    from x2paddle.optimizer.tf_optimizer import TFOptimizer
J
jiangjiajun 已提交
115 116

    print("Now translating model from tensorflow to paddle.")
117 118 119 120 121 122 123
    model = TFDecoder(model_path, define_input_shape=define_input_shape)
    if not without_data_format_optimization:
        mapper = TFOpMapper(model)
        optimizer = TFOptimizer(mapper)
        # neccesary optimization
        optimizer.delete_redundance_code()
        # optimizer below is experimental
J
jiangjiajun 已提交
124
        optimizer.optimize_elementwise_op()
125 126
        optimizer.merge_activation()
        optimizer.merge_bias()
J
jiangjiajun 已提交
127 128 129 130 131
        optimizer.optimize_sub_graph()


#        optimizer.merge_batch_norm()
#        optimizer.merge_prelu()
132 133 134 135
    else:
        mapper = TFOpMapperNHWC(model)
        optimizer = TFOptimizer(mapper)
        optimizer.delete_redundance_code()
J
jiangjiajun 已提交
136
        optimizer.strip_graph()
137 138
        optimizer.merge_activation()
        optimizer.merge_bias()
J
jiangjiajun 已提交
139
        optimizer.make_nchw_input_output()
140
        optimizer.remove_transpose()
M
mamingjie-China 已提交
141
    mapper.save_inference_model(save_dir, params_merge)
142 143


M
mamingjie-China 已提交
144
def caffe2paddle(proto, weight, save_dir, caffe_proto, params_merge=False):
J
jiangjiajun 已提交
145 146
    from x2paddle.decoder.caffe_decoder import CaffeDecoder
    from x2paddle.op_mapper.caffe_op_mapper import CaffeOpMapper
147
    from x2paddle.optimizer.caffe_optimizer import CaffeOptimizer
S
SunAhong1993 已提交
148
    import google.protobuf as gpb
S
SunAhong1993 已提交
149 150 151 152 153
    ver_part = gpb.__version__.split('.')
    version_satisfy = False
    if (int(ver_part[0]) == 3 and int(ver_part[1]) >= 6) \
        or (int(ver_part[0]) > 3):
        version_satisfy = True
J
jiangjiajun@baidu.com 已提交
154
    assert version_satisfy, '[ERROR] google.protobuf >= 3.6.0 is required'
J
jiangjiajun 已提交
155
    print("Now translating model from caffe to paddle.")
S
SunAhong1993 已提交
156
    model = CaffeDecoder(proto, weight, caffe_proto)
J
jiangjiajun 已提交
157
    mapper = CaffeOpMapper(model)
158 159 160
    optimizer = CaffeOptimizer(mapper)
    optimizer.merge_bn_scale()
    optimizer.merge_op_activation()
M
mamingjie-China 已提交
161
    mapper.save_inference_model(save_dir, params_merge)
162 163


M
mamingjie-China 已提交
164
def onnx2paddle(model_path, save_dir, params_merge=False):
C
update  
channingss 已提交
165 166 167 168
    # check onnx installation and version
    try:
        import onnx
        version = onnx.version.version
169
        if version != '1.6.0':
J
jiangjiajun@baidu.com 已提交
170
            print("[ERROR] onnx==1.6.0 is required")
C
update  
channingss 已提交
171 172
            return
    except:
J
jiangjiajun@baidu.com 已提交
173
        print("[ERROR] onnx is not installed, use \"pip install onnx==1.6.0\".")
C
update  
channingss 已提交
174
        return
C
channingss 已提交
175
    print("Now translating model from onnx to paddle.")
C
update  
channingss 已提交
176

R
root 已提交
177
    from x2paddle.op_mapper.onnx_op_mapper import ONNXOpMapper
C
update  
channingss 已提交
178
    from x2paddle.decoder.onnx_decoder import ONNXDecoder
R
root 已提交
179
    from x2paddle.optimizer.onnx_optimizer import ONNXOptimizer
180
    import onnxruntime
R
root 已提交
181
    model = ONNXDecoder(model_path)
C
channingss 已提交
182
    mapper = ONNXOpMapper(model, save_dir)
C
update  
channingss 已提交
183
    optimizer = ONNXOptimizer(mapper)
C
channingss 已提交
184

C
update  
channingss 已提交
185
    optimizer.delete_redundance_code()
M
mamingjie-China 已提交
186
    mapper.save_inference_model(save_dir, params_merge)
C
update  
channingss 已提交
187 188


J
jiangjiajun 已提交
189 190 191 192 193 194 195 196
def paddle2onnx(model_path, save_dir):
    from x2paddle.decoder.paddle_decoder import PaddleDecoder
    from x2paddle.op_mapper.paddle_op_mapper import PaddleOpMapper
    model = PaddleDecoder(model_path, '__model__', '__params__')
    mapper = PaddleOpMapper()
    mapper.convert(model.program, save_dir)


197
def main():
J
jiangjiajun 已提交
198
    if len(sys.argv) < 2:
C
update  
channingss 已提交
199
        print("Use \"x2paddle -h\" to print the help information")
J
jiangjiajun 已提交
200 201
        print("For more information, please follow our github repo below:)")
        print("\nGithub: https://github.com/PaddlePaddle/X2Paddle.git\n")
J
jiangjiajun 已提交
202 203
        return

204 205 206
    parser = arg_parser()
    args = parser.parse_args()

J
jiangjiajun 已提交
207
    if args.version:
J
jiangjiajun 已提交
208
        import x2paddle
M
mamingjie-China 已提交
209
        print("x2paddle-{} with python>=3.5, paddlepaddle>=1.6.0\n".format(
J
jiangjiajun 已提交
210
            x2paddle.__version__))
J
jiangjiajun 已提交
211 212
        return

J
Jason 已提交
213
    assert args.framework is not None, "--framework is not defined(support tensorflow/caffe/onnx)"
214
    assert args.save_dir is not None, "--save_dir is not defined"
M
mamingjie-China 已提交
215

R
root 已提交
216 217 218 219 220
    if args.framework == "onnx":
        try:
            import onnxruntime as rt
            version = rt.__version__
            if version != '1.0.0':
J
jiangjiajun@baidu.com 已提交
221
                print("[ERROR] onnxruntime==1.0.0 is required")
R
root 已提交
222 223
                return
        except:
M
mamingjie-China 已提交
224
            print(
J
jiangjiajun@baidu.com 已提交
225
                "[ERROR] onnxruntime is not installed, use \"pip install onnxruntime==1.0.0\"."
M
mamingjie-China 已提交
226 227
            )

M
mamingjie-China 已提交
228 229 230
    try:
        import paddle
        v0, v1, v2 = paddle.__version__.split('.')
J
jiangjiajun@baidu.com 已提交
231 232 233 234
        print("paddle.__version__ = {}".format(paddle.__version__))
        if v0 == '0' and v1 == '0' and v2 == '0':
            print("[WARNING] You are use develop version of paddlepaddle")
        elif int(v0) != 1 or int(v1) < 6:
J
jiangjiajun@baidu.com 已提交
235
            print("[ERROR] paddlepaddle>=1.6.0 is required")
M
mamingjie-China 已提交
236 237
            return
    except:
J
jiangjiajun@baidu.com 已提交
238 239 240
        print(
            "[ERROR] paddlepaddle not installed, use \"pip install paddlepaddle\""
        )
241 242

    if args.framework == "tensorflow":
J
jiangjiajun 已提交
243
        assert args.model is not None, "--model should be defined while translating tensorflow model"
244 245
        without_data_format_optimization = False
        define_input_shape = False
M
mamingjie-China 已提交
246
        params_merge = False
247 248 249 250
        if args.without_data_format_optimization:
            without_data_format_optimization = True
        if args.define_input_shape:
            define_input_shape = True
M
mamingjie-China 已提交
251 252
        if args.params_merge:
            params_merge = True
253
        tf2paddle(args.model, args.save_dir, without_data_format_optimization,
M
mamingjie-China 已提交
254
                  define_input_shape, params_merge)
255 256

    elif args.framework == "caffe":
S
SunAhong1993 已提交
257
        assert args.prototxt is not None and args.weight is not None, "--prototxt and --weight should be defined while translating caffe model"
M
mamingjie-China 已提交
258 259 260
        params_merge = False
        if args.params_merge:
            params_merge = True
S
SunAhong1993 已提交
261
        caffe2paddle(args.prototxt, args.weight, args.save_dir,
M
mamingjie-China 已提交
262
                     args.caffe_proto, params_merge)
C
update  
channingss 已提交
263 264
    elif args.framework == "onnx":
        assert args.model is not None, "--model should be defined while translating onnx model"
M
mamingjie-China 已提交
265 266 267 268
        params_merge = False
        if args.params_merge:
            params_merge = True
        onnx2paddle(args.model, args.save_dir, params_merge)
J
jiangjiajun 已提交
269 270 271 272 273

    elif args.framework == "paddle2onnx":
        assert args.model is not None, "--model should be defined while translating paddle model to onnx"
        paddle2onnx(args.model, args.save_dir)

274
    else:
J
jiangjiajun 已提交
275 276
        raise Exception(
            "--framework only support tensorflow/caffe/onnx/paddle2onnx now")
277 278 279 280


if __name__ == "__main__":
    main()