convert.py 8.4 KB
Newer Older
J
jiangjiajun 已提交
1 2 3 4 5 6 7 8 9 10 11 12 13
#   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.
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 22 23 24 25
def arg_parser():
    parser = argparse.ArgumentParser()
    parser.add_argument("--model",
                        "-m",
                        type=_text_type,
                        default=None,
J
upgrade  
jiangjiajun 已提交
26
                        help="define model file path for tensorflow or onnx")
S
SunAhong1993 已提交
27
    parser.add_argument("--prototxt",
28 29 30
                        "-p",
                        type=_text_type,
                        default=None,
S
SunAhong1993 已提交
31
                        help="prototxt file of caffe model")
32 33 34 35 36 37 38 39 40 41
    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 已提交
42 43 44 45 46 47
    parser.add_argument(
        "--framework",
        "-f",
        type=_text_type,
        default=None,
        help="define which deeplearning framework(tensorflow/caffe/onnx)")
S
SunAhong1993 已提交
48 49 50 51 52
    parser.add_argument(
        "--caffe_proto",
        "-c",
        type=_text_type,
        default=None,
J
upgrade  
jiangjiajun 已提交
53 54
        help="optional: the .py file compiled by caffe proto file of caffe model"
    )
J
jiangjiajun 已提交
55 56 57 58 59
    parser.add_argument("--version",
                        "-v",
                        action="store_true",
                        default=False,
                        help="get version of x2paddle")
60 61 62 63 64 65 66 67 68 69 70
    parser.add_argument(
        "--without_data_format_optimization",
        "-wo",
        action="store_true",
        default=False,
        help="tf model conversion without data format optimization")
    parser.add_argument("--define_input_shape",
                        "-d",
                        action="store_true",
                        default=False,
                        help="define input shape for tf model")
M
mamingjie-China 已提交
71 72 73 74 75
    parser.add_argument("--params_merge",
                        "-pm",
                        action="store_true",
                        default=False,
                        help="define whether merge the params")
J
jiangjiajun 已提交
76

77
    return parser
J
jiangjiajun 已提交
78

79

80 81 82
def tf2paddle(model_path,
              save_dir,
              without_data_format_optimization=False,
M
mamingjie-China 已提交
83 84
              define_input_shape=False,
              params_merge=False):
J
jiangjiajun 已提交
85 86
    # check tensorflow installation and version
    try:
87 88
        import os
        os.environ["TF_CPP_MIN_LOG_LEVEL"] = '3'
J
jiangjiajun 已提交
89 90 91 92 93 94 95 96
        import tensorflow as tf
        version = tf.__version__
        if version >= '2.0.0' or version < '1.0.0':
            print(
                "1.0.0<=tensorflow<2.0.0 is required, and v1.14.0 is recommended"
            )
            return
    except:
S
SunAhong1993 已提交
97
        print("Tensorflow is not installed, use \"pip install tensorflow\".")
J
jiangjiajun 已提交
98 99
        return

J
jiangjiajun 已提交
100
    from x2paddle.decoder.tf_decoder import TFDecoder
J
jiangjiajun 已提交
101
    from x2paddle.op_mapper.tf_op_mapper import TFOpMapper
102
    from x2paddle.op_mapper.tf_op_mapper_nhwc import TFOpMapperNHWC
J
jiangjiajun 已提交
103
    from x2paddle.optimizer.tf_optimizer import TFOptimizer
J
jiangjiajun 已提交
104 105

    print("Now translating model from tensorflow to paddle.")
106
    model = TFDecoder(model_path, define_input_shape=define_input_shape)
J
jiangjiajun 已提交
107 108 109 110 111 112
    mapper = TFOpMapperNHWC(model)
    optimizer = TFOptimizer(mapper)
    optimizer.delete_redundance_code()
    optimizer.strip_graph()
    #        optimizer.merge_activation()
    #        optimizer.merge_bias()
M
mamingjie-China 已提交
113
    mapper.save_inference_model(save_dir, params_merge)
114 115


M
mamingjie-China 已提交
116
def caffe2paddle(proto, weight, save_dir, caffe_proto, params_merge=False):
J
jiangjiajun 已提交
117 118
    from x2paddle.decoder.caffe_decoder import CaffeDecoder
    from x2paddle.op_mapper.caffe_op_mapper import CaffeOpMapper
119
    from x2paddle.optimizer.caffe_optimizer import CaffeOptimizer
S
SunAhong1993 已提交
120
    import google.protobuf as gpb
S
SunAhong1993 已提交
121 122 123 124 125
    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
S
SunAhong1993 已提交
126
    assert version_satisfy, 'google.protobuf >= 3.6.0 is required'
J
jiangjiajun 已提交
127
    print("Now translating model from caffe to paddle.")
S
SunAhong1993 已提交
128
    model = CaffeDecoder(proto, weight, caffe_proto)
J
jiangjiajun 已提交
129
    mapper = CaffeOpMapper(model)
130 131 132
    optimizer = CaffeOptimizer(mapper)
    optimizer.merge_bn_scale()
    optimizer.merge_op_activation()
M
mamingjie-China 已提交
133
    mapper.save_inference_model(save_dir, params_merge)
134 135


M
mamingjie-China 已提交
136
def onnx2paddle(model_path, save_dir, params_merge=False):
C
update  
channingss 已提交
137 138 139 140 141 142 143 144 145 146
    # check onnx installation and version
    try:
        import onnx
        version = onnx.version.version
        if version != '1.5.0':
            print("onnx==1.5.0 is required")
            return
    except:
        print("onnx is not installed, use \"pip install onnx==1.5.0\".")
        return
C
channingss 已提交
147
    print("Now translating model from onnx to paddle.")
C
update  
channingss 已提交
148 149

    from x2paddle.decoder.onnx_decoder import ONNXDecoder
C
channingss 已提交
150
    model = ONNXDecoder(model_path)
C
channingss 已提交
151 152

    from x2paddle.op_mapper.onnx_op_mapper import ONNXOpMapper
C
channingss 已提交
153
    mapper = ONNXOpMapper(model, save_dir)
C
channingss 已提交
154 155

    from x2paddle.optimizer.onnx_optimizer import ONNXOptimizer
C
update  
channingss 已提交
156
    optimizer = ONNXOptimizer(mapper)
C
channingss 已提交
157

C
update  
channingss 已提交
158
    optimizer.delete_redundance_code()
M
mamingjie-China 已提交
159
    mapper.save_inference_model(save_dir, params_merge)
C
update  
channingss 已提交
160 161


162
def main():
J
jiangjiajun 已提交
163
    if len(sys.argv) < 2:
C
update  
channingss 已提交
164
        print("Use \"x2paddle -h\" to print the help information")
J
jiangjiajun 已提交
165 166
        print("For more information, please follow our github repo below:)")
        print("\nGithub: https://github.com/PaddlePaddle/X2Paddle.git\n")
J
jiangjiajun 已提交
167 168
        return

169 170 171
    parser = arg_parser()
    args = parser.parse_args()

J
jiangjiajun 已提交
172
    if args.version:
J
jiangjiajun 已提交
173
        import x2paddle
J
jiangjiajun 已提交
174
        print("x2paddle-{} with python>=3.5, paddlepaddle>=1.6.1\n".format(
J
jiangjiajun 已提交
175
            x2paddle.__version__))
J
jiangjiajun 已提交
176 177
        return

J
jiangjiajun 已提交
178 179 180 181 182 183 184 185 186 187 188 189 190 191
    try:
        import paddle
        v0, v1, v2 = paddle.__version__.split('.')
        if int(v0) == 0 and int(v1) == 0 and int(v2) == 0:
            print(
                "You have installed paddlepaddle-dev? We're not sure it's working for x2paddle!"
            )
            print(
                "==================paddlepaddle>=1.6.1 is strongly recommended================="
            )
        elif int(v0) != 1 or int(v1) < 6:
            print("paddlepaddle>=1.6.1 is required")
            return
    except:
J
jiangjiajun 已提交
192
        print("paddlepaddle not installed, use \"pip install paddlepaddle\"")
J
jiangjiajun 已提交
193
        return
J
jiangjiajun 已提交
194

J
Jason 已提交
195
    assert args.framework is not None, "--framework is not defined(support tensorflow/caffe/onnx)"
196 197 198
    assert args.save_dir is not None, "--save_dir is not defined"

    if args.framework == "tensorflow":
J
jiangjiajun 已提交
199
        assert args.model is not None, "--model should be defined while translating tensorflow model"
200 201
        without_data_format_optimization = False
        define_input_shape = False
M
mamingjie-China 已提交
202
        params_merge = False
203 204 205 206
        if args.without_data_format_optimization:
            without_data_format_optimization = True
        if args.define_input_shape:
            define_input_shape = True
M
mamingjie-China 已提交
207 208
        if args.params_merge:
            params_merge = True
209
        tf2paddle(args.model, args.save_dir, without_data_format_optimization,
M
mamingjie-China 已提交
210
                  define_input_shape, params_merge)
211 212

    elif args.framework == "caffe":
S
SunAhong1993 已提交
213
        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 已提交
214 215 216
        params_merge = False
        if args.params_merge:
            params_merge = True
S
SunAhong1993 已提交
217
        caffe2paddle(args.prototxt, args.weight, args.save_dir,
M
mamingjie-China 已提交
218
                     args.caffe_proto, params_merge)
C
update  
channingss 已提交
219 220
    elif args.framework == "onnx":
        assert args.model is not None, "--model should be defined while translating onnx model"
M
mamingjie-China 已提交
221 222 223 224
        params_merge = False
        if args.params_merge:
            params_merge = True
        onnx2paddle(args.model, args.save_dir, params_merge)
225
    else:
C
update  
channingss 已提交
226
        raise Exception("--framework only support tensorflow/caffe/onnx now")
227 228 229 230


if __name__ == "__main__":
    main()