1 /* Copyright 2018 The TensorFlow Authors. All Rights Reserved.
2 
3 Licensed under the Apache License, Version 2.0 (the "License");
4 you may not use this file except in compliance with the License.
5 You may obtain a copy of the License at
6 
7     http://www.apache.org/licenses/LICENSE-2.0
8 
9 Unless required by applicable law or agreed to in writing, software
10 distributed under the License is distributed on an "AS IS" BASIS,
11 WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
12 See the License for the specific language governing permissions and
13 limitations under the License.
14 ==============================================================================*/
15 
16 #include "tensorflow/js/ops/ts_op_gen.h"
17 #include <unordered_map>
18 
19 #include "tensorflow/core/framework/api_def.pb.h"
20 #include "tensorflow/core/framework/op_def_util.h"
21 #include "tensorflow/core/lib/gtl/map_util.h"
22 #include "tensorflow/core/lib/strings/strcat.h"
23 #include "tensorflow/core/public/version.h"
24 
25 namespace tensorflow {
26 namespace {
27 
IsListAttr(const OpDef_ArgDef & arg)28 static bool IsListAttr(const OpDef_ArgDef& arg) {
29   return !arg.type_list_attr().empty() || !arg.number_attr().empty();
30 }
31 
32 // Struct to hold a combo OpDef and ArgDef for a given Op argument:
33 struct ArgDefs {
ArgDefstensorflow::__anon850744f40111::ArgDefs34   ArgDefs(const OpDef::ArgDef& op_def_arg, const ApiDef::Arg& api_def_arg)
35       : op_def_arg(op_def_arg), api_def_arg(api_def_arg) {}
36 
37   const OpDef::ArgDef& op_def_arg;
38   const ApiDef::Arg& api_def_arg;
39 };
40 
41 // Struct to hold a combo OpDef::AttrDef and ApiDef::Attr for an Op.
42 struct OpAttrs {
OpAttrstensorflow::__anon850744f40111::OpAttrs43   OpAttrs(const OpDef::AttrDef& op_def_attr, const ApiDef::Attr& api_def_attr)
44       : op_def_attr(op_def_attr), api_def_attr(api_def_attr) {}
45 
46   const OpDef::AttrDef& op_def_attr;
47   const ApiDef::Attr& api_def_attr;
48 };
49 
50 // Helper class to generate TypeScript code for a given OpDef:
51 class GenTypeScriptOp {
52  public:
53   GenTypeScriptOp(const OpDef& op_def, const ApiDef& api_def);
54   ~GenTypeScriptOp();
55 
56   // Returns the generated code as a string:
57   string Code();
58 
59  private:
60   void ProcessArgs();
61   void ProcessAttrs();
62   void AddAttrForArg(const string& attr, int arg_index);
63   string InputForAttr(const OpDef::AttrDef& op_def_attr);
64 
65   void AddMethodSignature();
66   void AddOpAttrs();
67   void AddMethodReturnAndClose();
68 
69   const OpDef& op_def_;
70   const ApiDef& api_def_;
71 
72   // Placeholder string for all generated code:
73   string result_;
74 
75   // Holds in-order vector of Op inputs:
76   std::vector<ArgDefs> input_op_args_;
77 
78   // Holds in-order vector of Op attributes:
79   std::vector<OpAttrs> op_attrs_;
80 
81   // Stores attributes-to-arguments by name:
82   typedef std::unordered_map<string, std::vector<int>> AttrArgIdxMap;
83   AttrArgIdxMap attr_arg_idx_map_;
84 
85   // Holds number of outputs:
86   int num_outputs_;
87 };
88 
GenTypeScriptOp(const OpDef & op_def,const ApiDef & api_def)89 GenTypeScriptOp::GenTypeScriptOp(const OpDef& op_def, const ApiDef& api_def)
90     : op_def_(op_def), api_def_(api_def), num_outputs_(0) {}
91 
~GenTypeScriptOp()92 GenTypeScriptOp::~GenTypeScriptOp() {}
93 
Code()94 string GenTypeScriptOp::Code() {
95   ProcessArgs();
96   ProcessAttrs();
97 
98   // Generate exported function for Op:
99   AddMethodSignature();
100   AddOpAttrs();
101   AddMethodReturnAndClose();
102 
103   strings::StrAppend(&result_, "\n");
104   return result_;
105 }
106 
ProcessArgs()107 void GenTypeScriptOp::ProcessArgs() {
108   for (int i = 0; i < api_def_.arg_order_size(); i++) {
109     auto op_def_arg = FindInputArg(api_def_.arg_order(i), op_def_);
110     if (op_def_arg == nullptr) {
111       LOG(WARNING) << "Could not find OpDef::ArgDef for "
112                    << api_def_.arg_order(i);
113       continue;
114     }
115     auto api_def_arg = FindInputArg(api_def_.arg_order(i), api_def_);
116     if (api_def_arg == nullptr) {
117       LOG(WARNING) << "Could not find ApiDef::Arg for "
118                    << api_def_.arg_order(i);
119       continue;
120     }
121 
122     // Map attr names to arg indexes:
123     if (!op_def_arg->type_attr().empty()) {
124       AddAttrForArg(op_def_arg->type_attr(), i);
125     } else if (!op_def_arg->type_list_attr().empty()) {
126       AddAttrForArg(op_def_arg->type_list_attr(), i);
127     }
128     if (!op_def_arg->number_attr().empty()) {
129       AddAttrForArg(op_def_arg->number_attr(), i);
130     }
131 
132     input_op_args_.push_back(ArgDefs(*op_def_arg, *api_def_arg));
133   }
134 
135   num_outputs_ = api_def_.out_arg_size();
136 }
137 
ProcessAttrs()138 void GenTypeScriptOp::ProcessAttrs() {
139   for (int i = 0; i < op_def_.attr_size(); i++) {
140     op_attrs_.push_back(OpAttrs(op_def_.attr(i), api_def_.attr(i)));
141   }
142 }
143 
AddAttrForArg(const string & attr,int arg_index)144 void GenTypeScriptOp::AddAttrForArg(const string& attr, int arg_index) {
145   // Keep track of attributes-to-arguments by name. These will be used for
146   // construction Op attributes that require information about the inputs.
147   auto iter = attr_arg_idx_map_.find(attr);
148   if (iter == attr_arg_idx_map_.end()) {
149     attr_arg_idx_map_.insert(AttrArgIdxMap::value_type(attr, {arg_index}));
150   } else {
151     iter->second.push_back(arg_index);
152   }
153 }
154 
InputForAttr(const OpDef::AttrDef & op_def_attr)155 string GenTypeScriptOp::InputForAttr(const OpDef::AttrDef& op_def_attr) {
156   string inputs;
157   auto arg_list = attr_arg_idx_map_.find(op_def_attr.name());
158   if (arg_list != attr_arg_idx_map_.end()) {
159     for (auto iter = arg_list->second.begin(); iter != arg_list->second.end();
160          ++iter) {
161       strings::StrAppend(&inputs, input_op_args_[*iter].op_def_arg.name());
162     }
163   }
164   return inputs;
165 }
166 
AddMethodSignature()167 void GenTypeScriptOp::AddMethodSignature() {
168   strings::StrAppend(&result_, "export function ", api_def_.endpoint(0).name(),
169                      "(");
170 
171   bool is_first = true;
172   for (auto& in_arg : input_op_args_) {
173     if (is_first) {
174       is_first = false;
175     } else {
176       strings::StrAppend(&result_, ", ");
177     }
178 
179     auto op_def_arg = in_arg.op_def_arg;
180 
181     strings::StrAppend(&result_, op_def_arg.name(), ": ");
182     if (IsListAttr(op_def_arg)) {
183       strings::StrAppend(&result_, "tfc.Tensor[]");
184     } else {
185       strings::StrAppend(&result_, "tfc.Tensor");
186     }
187   }
188 
189   if (num_outputs_ == 1) {
190     strings::StrAppend(&result_, "): tfc.Tensor {\n");
191   } else {
192     strings::StrAppend(&result_, "): tfc.Tensor[] {\n");
193   }
194 }
195 
AddOpAttrs()196 void GenTypeScriptOp::AddOpAttrs() {
197   strings::StrAppend(&result_, "  const opAttrs = [\n");
198 
199   bool is_first = true;
200   for (auto& attr : op_attrs_) {
201     if (is_first) {
202       is_first = false;
203     } else {
204       strings::StrAppend(&result_, ",\n");
205     }
206 
207     // Append 4 spaces to start:
208     strings::StrAppend(&result_, "    ");
209 
210     if (attr.op_def_attr.type() == "type") {
211       // Type OpAttributes can be generated from a helper function:
212       strings::StrAppend(&result_, "createTensorsTypeOpAttr('",
213                          attr.op_def_attr.name(), "', ",
214                          InputForAttr(attr.op_def_attr), ")");
215     } else if (attr.op_def_attr.type() == "int") {
216       strings::StrAppend(&result_, "{name: '", attr.op_def_attr.name(), "', ");
217       strings::StrAppend(&result_, "type: nodeBackend().binding.TF_ATTR_INT, ");
218       strings::StrAppend(&result_, "value: ", InputForAttr(attr.op_def_attr),
219                          ".length}");
220     }
221   }
222   strings::StrAppend(&result_, "\n  ];\n");
223 }
224 
AddMethodReturnAndClose()225 void GenTypeScriptOp::AddMethodReturnAndClose() {
226   strings::StrAppend(&result_, "  return null;\n}\n");
227 }
228 
WriteTSOp(const OpDef & op_def,const ApiDef & api_def,WritableFile * ts)229 void WriteTSOp(const OpDef& op_def, const ApiDef& api_def, WritableFile* ts) {
230   GenTypeScriptOp ts_op(op_def, api_def);
231   TF_CHECK_OK(ts->Append(GenTypeScriptOp(op_def, api_def).Code()));
232 }
233 
StartFile(WritableFile * ts_file)234 void StartFile(WritableFile* ts_file) {
235   const string header =
236       R"header(/**
237  * @license
238  * Copyright 2018 Google Inc. All Rights Reserved.
239  * Licensed under the Apache License, Version 2.0 (the "License");
240  * you may not use this file except in compliance with the License.
241  * You may obtain a copy of the License at
242  *
243  * http://www.apache.org/licenses/LICENSE-2.0
244  *
245  * Unless required by applicable law or agreed to in writing, software
246  * distributed under the License is distributed on an "AS IS" BASIS,
247  * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
248  * See the License for the specific language governing permissions and
249  * limitations under the License.
250  * =============================================================================
251  */
252 
253 // This file is MACHINE GENERATED! Do not edit
254 
255 import * as tfc from '@tensorflow/tfjs-core';
256 import {createTensorsTypeOpAttr, nodeBackend} from './op_utils';
257 
258 )header";
259 
260   TF_CHECK_OK(ts_file->Append(header));
261 }
262 
263 }  // namespace
264 
WriteTSOps(const OpList & ops,const ApiDefMap & api_def_map,const string & ts_filename)265 void WriteTSOps(const OpList& ops, const ApiDefMap& api_def_map,
266                 const string& ts_filename) {
267   Env* env = Env::Default();
268 
269   std::unique_ptr<WritableFile> ts_file = nullptr;
270   TF_CHECK_OK(env->NewWritableFile(ts_filename, &ts_file));
271 
272   StartFile(ts_file.get());
273 
274   for (const auto& op_def : ops.op()) {
275     // Skip deprecated ops
276     if (op_def.has_deprecation() &&
277         op_def.deprecation().version() <= TF_GRAPH_DEF_VERSION) {
278       continue;
279     }
280 
281     const auto* api_def = api_def_map.GetApiDef(op_def.name());
282     if (api_def->visibility() == ApiDef::VISIBLE) {
283       WriteTSOp(op_def, *api_def, ts_file.get());
284     }
285   }
286 
287   TF_CHECK_OK(ts_file->Close());
288 }
289 
290 }  // namespace tensorflow
291