ir.cc 7.4 KB
Newer Older
F
flame 已提交
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15
// Copyright (c) 2018 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.

#include "paddle/fluid/pybind/ir.h"
W
WangZhen 已提交
16
#include <algorithm>
17
#include <memory>
F
flame 已提交
18 19
#include <string>
#include <unordered_map>
W
WangZhen 已提交
20
#include <unordered_set>
21
#include <utility>
F
flame 已提交
22
#include "paddle/fluid/framework/ir/graph.h"
W
WangZhen 已提交
23
#include "paddle/fluid/framework/ir/graph_helper.h"
W
WangZhen 已提交
24
#include "paddle/fluid/framework/ir/graph_pattern_detector.h"
F
flame 已提交
25 26
#include "paddle/fluid/framework/ir/node.h"
#include "paddle/fluid/framework/op_desc.h"
27
#include "paddle/fluid/framework/scope.h"
F
flame 已提交
28 29 30 31 32 33
#include "paddle/fluid/framework/var_desc.h"
#include "pybind11/stl.h"

namespace py = pybind11;
using paddle::framework::ir::Graph;
using paddle::framework::ir::Node;
34
using paddle::framework::ir::NodeComp;
W
WangZhen 已提交
35
using paddle::framework::ir::GraphSafeRemoveNodes;
W
WangZhen 已提交
36 37 38 39
using paddle::framework::ir::HasCircle;
using paddle::framework::ir::GraphNum;
using paddle::framework::ir::TopologySortOperations;
using paddle::framework::ir::BuildOperationAdjList;
F
flame 已提交
40 41
using paddle::framework::OpDesc;
using paddle::framework::ProgramDesc;
42
using paddle::framework::Scope;
F
flame 已提交
43 44 45 46 47 48
using paddle::framework::VarDesc;
using pybind11::return_value_policy;

namespace paddle {
namespace pybind {
void BindGraph(py::module *m) {
W
WangZhen 已提交
49
  m->def("graph_safe_remove_nodes", GraphSafeRemoveNodes);
W
WangZhen 已提交
50 51 52 53
  m->def("has_circle", HasCircle);
  m->def("graph_num", GraphNum);
  m->def("topology_sort", TopologySortOperations,
         return_value_policy::reference);
54
  m->def("build_adjacency_list", BuildOperationAdjList<NodeComp>,
W
WangZhen 已提交
55
         return_value_policy::reference);
F
flame 已提交
56 57 58 59 60
  py::class_<Graph, std::shared_ptr<Graph>>(
      *m, "Graph",
      "The graph is a Directed Acyclic Single Static Assignment Graph, see "
      "`paddle::ir::Graph` for details.")
      .def(py::init<const ProgramDesc &>())
61
      .def("clone", &Graph::Clone)
F
flame 已提交
62
      .def("has", &Graph::Has)
63
      .def("get_bool", &Graph::Get<bool>)
F
flame 已提交
64 65 66 67
      .def("get_int", &Graph::Get<int>)
      .def("get_float", &Graph::Get<float>)
      .def("get_double", &Graph::Get<double>)
      .def("get_string", &Graph::Get<std::string>)
68 69
      .def("get_marked_nodes", &Graph::Get<std::unordered_set<const Node *>>,
           return_value_policy::reference)
70 71
      .def("set", [](Graph &self, const std::string &attr_name,
                     bool attr) { return self.Set(attr_name, new bool(attr)); })
F
flame 已提交
72 73 74 75 76 77 78 79 80 81 82 83 84 85 86
      .def("set", [](Graph &self, const std::string &attr_name,
                     int attr) { return self.Set(attr_name, new int(attr)); })
      .def("set",
           [](Graph &self, const std::string &attr_name,
              const std::string &attr) {
             return self.Set(attr_name, new std::string(attr));
           })
      .def("set",
           [](Graph &self, const std::string &attr_name, float attr) {
             return self.Set(attr_name, new float(attr));
           })
      .def("set",
           [](Graph &self, const std::string &attr_name, double attr) {
             return self.Set(attr_name, new double(attr));
           })
W
WangZhen 已提交
87 88 89 90 91 92
      .def("set",
           [](Graph &self, const std::string &attr_name,
              const std::unordered_set<const Node *> &attr) {
             return self.Set(attr_name,
                             new std::unordered_set<const Node *>(attr));
           })
Z
Zeng Jinle 已提交
93 94 95 96 97 98
      .def("set",
           [](Graph &self, const std::string &attr_name,
              const std::unordered_set<std::string> &attr) {
             return self.Set(attr_name,
                             new std::unordered_set<std::string>(attr));
           })
99 100 101 102
      .def("set_not_owned",
           [](Graph &self, const std::string &attr_name, Scope &attr) {
             self.SetNotOwned<Scope>(attr_name, &attr);
           })
F
flame 已提交
103 104 105 106 107 108 109 110 111 112 113 114 115 116 117 118 119 120 121 122 123
      .def("erase", &Graph::Erase)
      .def("nodes", &Graph::Nodes, return_value_policy::reference)
      .def("create_var_node",
           [](Graph &self, VarDesc &var_desc) {
             return self.CreateVarNode(&var_desc);
           },
           return_value_policy::reference)
      .def("create_op_node",
           [](Graph &self, OpDesc &op_desc) {
             return self.CreateOpNode(&op_desc);
           },
           return_value_policy::reference)
      .def("create_control_dep_var", &Graph::CreateControlDepVar,
           return_value_policy::reference)
      .def("create_empty_node", &Graph::CreateEmptyNode,
           return_value_policy::reference)
      .def("release_nodes", &Graph::ReleaseNodes)
      .def("remove_node",
           [](Graph &self, Node &node) { return self.RemoveNode(&node); })
      .def("retrieve_node", &Graph::RetrieveNode,
           return_value_policy::reference)
X
Xin Pan 已提交
124
      .def("resolve_hazard", &Graph::ResolveHazard)
125 126
      .def("origin_program_desc", &Graph::OriginProgram,
           return_value_policy::reference);
F
flame 已提交
127 128 129 130 131 132
}

void BindNode(py::module *m) {
  py::class_<Node> node(*m, "Node");
  node.def("name", &Node::Name)
      .def("node_type", &Node::NodeType)
W
WangZhen 已提交
133 134
      .def("var", &Node::Var, return_value_policy::reference)
      .def("op", &Node::Op, return_value_policy::reference)
F
flame 已提交
135 136 137 138
      .def("id", &Node::id)
      .def("is_op", &Node::IsOp)
      .def("is_var", &Node::IsVar)
      .def("is_ctrl_var", &Node::IsCtrlVar)
W
WangZhen 已提交
139
      .def("clear_inputs", [](Node &self) { self.inputs.clear(); })
140
      .def("remove_input",
W
WangZhen 已提交
141
           [](Node &self, int node_id) {
W
WangZhen 已提交
142 143 144 145 146
             auto pos = std::find_if(
                 self.inputs.begin(), self.inputs.end(),
                 [&node_id](const Node *n) { return n->id() == node_id; });
             if (pos != self.inputs.end()) {
               self.inputs.erase(pos);
W
WangZhen 已提交
147 148
             }
           })
149
      .def("remove_input",
W
WangZhen 已提交
150
           [](Node &self, Node &node) {
W
WangZhen 已提交
151 152 153 154
             auto pos =
                 std::find(self.inputs.begin(), self.inputs.end(), &node);
             if (pos != self.inputs.end()) {
               self.inputs.erase(pos);
W
WangZhen 已提交
155 156
             }
           })
157
      .def("append_input",
W
WangZhen 已提交
158
           [](Node &self, Node &node) { self.inputs.push_back(&node); })
W
WangZhen 已提交
159
      .def("clear_outputs", [](Node &self) { self.outputs.clear(); })
160
      .def("remove_output",
W
WangZhen 已提交
161
           [](Node &self, int node_id) {
W
WangZhen 已提交
162 163 164 165 166
             auto pos = std::find_if(
                 self.outputs.begin(), self.outputs.end(),
                 [&node_id](const Node *n) { return n->id() == node_id; });
             if (pos != self.outputs.end()) {
               self.outputs.erase(pos);
W
WangZhen 已提交
167 168
             }
           })
169
      .def("remove_output",
W
WangZhen 已提交
170
           [](Node &self, Node &node) {
W
WangZhen 已提交
171 172 173 174
             auto pos =
                 std::find(self.outputs.begin(), self.outputs.end(), &node);
             if (pos != self.outputs.end()) {
               self.outputs.erase(pos);
W
WangZhen 已提交
175 176
             }
           })
177
      .def("append_output",
W
WangZhen 已提交
178
           [](Node &self, Node &node) { self.outputs.push_back(&node); })
179 180
      .def_readwrite("inputs", &Node::inputs)
      .def_readwrite("outputs", &Node::outputs);
F
flame 已提交
181 182 183 184 185 186 187 188

  py::enum_<Node::Type>(node, "Type")
      .value("Operation", Node::Type::kOperation)
      .value("Variable", Node::Type::kVariable)
      .export_values();
}
}  // namespace pybind
}  // namespace paddle