call_transformer.py 3.0 KB
Newer Older
1 2 3 4 5 6 7 8 9 10 11 12 13 14
#   Copyright (c) 2020 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
from paddle.utils import gast
16

17 18 19
from paddle.fluid.dygraph.dygraph_to_static.static_analysis import (
    AstNodeWrapper,
)
20 21
from paddle.fluid.dygraph.dygraph_to_static.utils import ast_to_source_code
from paddle.fluid.dygraph.dygraph_to_static.utils import is_paddle_api
22
from .base_transformer import (
23 24
    BaseTransformer,
)
25

26 27
PDB_SET = "pdb.set_trace"

28

29
class CallTransformer(BaseTransformer):
30 31 32 33 34 35 36 37 38 39 40
    """
    This class transforms function calls into Static Graph Ast.
    """

    def __init__(self, wrapper_root):
        assert isinstance(
            wrapper_root, AstNodeWrapper
        ), "Input non-AstNodeWrapper node for the initialization of CallTransformer."
        self.wrapper_root = wrapper_root
        self.root = wrapper_root.node

41 42 43 44 45
    def _no_need_convert_call(self, node):
        """
        Determines whether a function needs to be transformed by `convert_call`.
        It doesn't need to be transformed when a function satisfies the following conditions:
          1. It's a api of paddle
46
          2. It's a python builtin function not include `len`, `zip`, `range` and `enumerate`
47
        """
L
liym27 已提交
48
        assert isinstance(node, gast.Call)
49 50 51
        if is_paddle_api(node):
            return True

L
liym27 已提交
52 53
        func_str = ast_to_source_code(node.func).strip()
        try:
54
            from paddle.jit.dy2static.convert_call_func import (
55 56 57
                is_builtin,
            )

58 59 60 61 62 63
            need_convert_builtin_func_list = {
                'len',
                'zip',
                'range',
                'enumerate',
            }
64
            is_builtin = eval("is_builtin({})".format(func_str))
65 66
            need_convert = func_str in need_convert_builtin_func_list
            return is_builtin and not need_convert
L
liym27 已提交
67 68 69
        except Exception:
            return False

70 71 72 73 74
    def transform(self):
        self.visit(self.root)

    def visit_Call(self, node):
        self.generic_visit(node)
L
liym27 已提交
75

76
        if self._no_need_convert_call(node):
L
liym27 已提交
77 78
            return node

79
        func_str = ast_to_source_code(node.func).strip()
80 81 82 83 84 85

        # NOTE(liym27): Don't convert `pad.set_trace` even if the convertion doesn't work finally, because
        # it is clearer to see where it is called from.
        if PDB_SET in func_str:
            return node

86
        new_func_str = "_jst.Call({})".format(func_str)
87 88 89 90
        new_func_ast = gast.parse(new_func_str).body[0].value
        node.func = new_func_ast

        return node