forked from pytorch/pytorch
-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathbailout_graph.cpp
396 lines (351 loc) · 13.4 KB
/
bailout_graph.cpp
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
248
249
250
251
252
253
254
255
256
257
258
259
260
261
262
263
264
265
266
267
268
269
270
271
272
273
274
275
276
277
278
279
280
281
282
283
284
285
286
287
288
289
290
291
292
293
294
295
296
297
298
299
300
301
302
303
304
305
306
307
308
309
310
311
312
313
314
315
316
317
318
319
320
321
322
323
324
325
326
327
328
329
330
331
332
333
334
335
336
337
338
339
340
341
342
343
344
345
346
347
348
349
350
351
352
353
354
355
356
357
358
359
360
361
362
363
364
365
366
367
368
369
370
371
372
373
374
375
376
377
378
379
380
381
382
383
384
385
386
387
388
389
390
391
392
393
394
395
396
#include <torch/csrc/jit/passes/bailout_graph.h>
#include <ATen/core/function.h>
#include <c10/util/irange.h>
#include <torch/csrc/jit/ir/alias_analysis.h>
#include <torch/csrc/jit/ir/ir_views.h>
#include <torch/csrc/jit/jit_log.h>
#include <torch/csrc/jit/passes/clear_profiling.h>
#include <torch/csrc/jit/passes/constant_pooling.h>
#include <torch/csrc/jit/passes/liveness.h>
#include <memory>
#include <unordered_set>
#include <utility>
namespace torch::jit {
static bool shouldBeCapturedInByBailOut(Node* n) {
return n->kind() != prim::Constant;
}
struct BailOutGraphBuilderForNode {
explicit BailOutGraphBuilderForNode(
std::shared_ptr<Graph> graph,
std::shared_ptr<Graph> target)
: graph_(std::move(graph)), copy_graph_(std::move(target)) {}
// capture `old_value` into the bailout graph
// by creating a new input and mapping
// `old_value` to it
Value* addNewInputForValue(Value* old_value) {
auto node = old_value->node();
// this reduces the number of inputs to a bailout graph significantly
// making it easier to debug
if (node->kind() == prim::Constant) {
TORCH_INTERNAL_ASSERT(!shouldBeCapturedInByBailOut(node));
auto new_const = copy_graph_->createClone(node, {nullptr});
copy_graph_->block()->prependNode(new_const);
return new_const->output();
}
live_inputs_.push_back(old_value);
auto new_value = copy_graph_->block()->addInput();
GRAPH_DEBUG(
"Adding a new value %",
new_value->debugName(),
" for %",
old_value->debugName());
return mapValueAndCopyMetadata(old_value, new_value);
}
Value* mapValueAndCopyMetadata(Value* old_value, Value* new_value) {
this->old_to_new_[old_value] = new_value;
new_value->copyMetadata(old_value);
return new_value;
}
Value* getOrAddInputForValue(Value* v) {
if (this->old_to_new_.count(v) == 0) {
return addNewInputForValue(v);
} else {
return this->old_to_new_[v];
}
}
Value* getInputForValue(Value* v) {
TORCH_INTERNAL_ASSERT(this->old_to_new_.count(v));
return this->old_to_new_[v];
}
Node* cloneNode(Node* node) {
auto* block = copy_graph_->block();
auto env = [this](Value* v) { return getOrAddInputForValue(v); };
auto new_node = block->appendNode(copy_graph_->createClone(node, env));
for (size_t i = 0; i < node->outputs().size(); ++i) {
auto oo = node->outputs()[i];
auto no = new_node->outputs()[i];
old_to_new_[oo] = no;
}
return new_node;
}
// buildBailOutBlockFrom builds a bailout graph from
// a given node `n` until the end of the owning block
// If `n` belongs to `prim::If` or `prim::Loop`
// buildBailOutLoop/If continue
// from block's owning node (e.g. `prim::If` or
// `prim::Loop`)
void buildBailOutBlockFrom(Node* n) {
auto b = n->owningBlock();
for (auto it = n->iterator(); it != b->nodes().end(); it++) {
cloneNode(*it);
}
// we are either in `prim::If` or `prim::Loop`
// bailout graph building will continue from `outer_node` next
auto outer_node = n->owningBlock()->owningNode();
if (outer_node) {
if (outer_node->kind() == prim::Loop) {
buildBailOutLoop(outer_node);
} else if (outer_node->kind() == prim::If) {
buildBailOutIf(b->outputs(), outer_node);
} else {
AT_ERROR("Unexpected outer node");
}
}
}
void mapValues(
const at::ArrayRef<Value*> block_outputs,
const at::ArrayRef<Value*> carried_deps) {
TORCH_INTERNAL_ASSERT(block_outputs.size() == carried_deps.size());
for (const auto i : c10::irange(block_outputs.size())) {
auto nv = getOrAddInputForValue(block_outputs[i]);
old_to_new_[carried_deps[i]] = nv;
}
}
void buildBailOutLoop(Node* outer_node) {
LoopView lv(outer_node);
auto old_max_count = getOrAddInputForValue(lv.maxTripCount());
auto cur_iter = getInputForValue(lv.currentTripCount());
auto block_outputs = lv.bodyBlock()->outputs();
auto* block = copy_graph_->block();
// subtract the number of iterations
WithInsertPoint guard(*block->nodes().end());
auto updated_max_trip_count =
copy_graph_->insert(aten::sub, {old_max_count, cur_iter});
auto one = copy_graph_->insertConstant({1});
updated_max_trip_count =
copy_graph_->insert(aten::sub, {updated_max_trip_count, one});
auto cur_plus_one = copy_graph_->insert(aten::add, {one, cur_iter});
// We need to be careful when mapping `block_outputs` to continuation
// loop's inputs since `cloneFrom` will replace `%4` with the same value
// in both, `prim::Loop` and `aten::cat` in the example below:
//
// ... : Tensor = prim::Loop(%MAX_TRIP_COUNT, %COND, ..., %4)
// block0(%i.2 : int, ...):
// ...
// %y.5 : Double(3) = aten::cat(%22, %4)
// ...
//
// However for the cloned loop node, the values should be different.
// Namely, the value in `prim::Loop` should come from
// `lv.bodyBlock()->outputs()` which are mapped to the outputs of the
// current iteration whereas `%4` in `aten::cat` needs to be mapped to the
// cloned value of `%4` in a bailout graph. To work around this, we manually
// clone loop nodes
// map the residual loop's inputs to the outputs of the current iteration
// (i.e. `block_outputs`)
auto new_loop =
copy_graph_->insertNode(copy_graph_->create(prim::Loop, {}, 0))
->setSourceRange(outer_node->sourceRange());
new_loop->addInput(updated_max_trip_count);
for (auto bo : block_outputs) {
new_loop->addInput(getOrAddInputForValue(bo));
}
// clone the loop body and map old loop's outputs to new loop's outputs
auto new_loop_body = new_loop->addBlock();
auto env = [this](Value* v) { return getOrAddInputForValue(v); };
new_loop_body->cloneFrom(lv.bodyBlock(), env);
for (auto ov : lv.carriedOutputs()) {
auto no = new_loop->addOutput();
mapValueAndCopyMetadata(ov, no);
}
LoopView new_lv(new_loop);
{
WithInsertPoint guard_in_loop(*new_lv.bodyBlock()->nodes().begin());
// `one` will be replaced with new_lv.currentTripCount()
// but it needs to be done after
// new_lv.currentTripCount()->replaceAllUsesWith(adj_iter_ctr);
// to avoid cyclical references
auto adj_iter_ctr = copy_graph_->insert(aten::add, {cur_plus_one, one});
new_lv.currentTripCount()->replaceAllUsesWith(adj_iter_ctr);
adj_iter_ctr->node()->replaceInputWith(one, new_lv.currentTripCount());
}
if (outer_node->next()) {
buildBailOutBlockFrom(outer_node->next());
}
}
void buildBailOutIf(
const at::ArrayRef<Value*> block_outputs,
Node* outer_node) {
auto if_outputs = outer_node->outputs();
mapValues(block_outputs, if_outputs);
buildBailOutBlockFrom(outer_node->next());
}
std::shared_ptr<Graph> buildBailOutGraphFrom(Node* n) {
// add graph inputs for guard's input
// and loop counts for loops `n` is contained in
// to make sure we can line bailout grap's inputs up properly
// with arguments to this BailOut node.
for (auto bi : n->inputs()) {
getOrAddInputForValue(bi);
}
buildBailOutBlockFrom(n);
// add graph outputs
for (auto ov : graph_->outputs()) {
copy_graph_->registerOutput(getOrAddInputForValue(ov));
}
return copy_graph_;
}
std::shared_ptr<Graph> graph_;
std::shared_ptr<Graph> copy_graph_;
std::vector<Value*> live_inputs_;
std::unordered_map<Value*, Value*> old_to_new_;
};
// `BailOutInserter` replaces prim::Guard nodes with
// prim::BailOut nodes that allow interpreter to
// resume execution of the unoptimized(deoptimized)
// version of an original graph from a particular point
struct BailOutInserter {
explicit BailOutInserter(std::shared_ptr<Graph> graph)
: graph_(std::move(graph)) {}
void run() {
liveness_sets_ = BuildLivenessSets(graph_);
insertBailOuts(graph_->block());
replaceGuardsWithBailouts();
// embed a full original graph
addUnoptimizedFuncToBailouts();
}
// Packs the original unoptimized graph into a Function constant
// and add it as the first input to every prim::BailOut point
// This graph will be used to compute a bailout graph for
// any given bailout point
void addUnoptimizedFuncToBailouts() {
auto unoptimized_graph = graph_->copy();
auto unopt_func = graph_->create(prim::BailoutTemplate)
->insertAfter(graph_->param_node());
// Returns an int so that we have an easy way to do graph traversal
unopt_func->output()->setType(IntType::get());
unopt_func->g_(attr::Subgraph, std::move(unoptimized_graph));
for (auto bn : bailouts_) {
bn->insertInput(0, unopt_func->output());
}
}
// Removes guards by hooking up the guarded tensor
// directly to its users and also clears
// profiling information on it.
void removeGuards(Block* b) {
for (auto it = b->nodes().begin(); it != b->nodes().end(); ++it) {
if (it->kind() == prim::Guard) {
// this will need to be profiled again
it->input()->setType(TensorType::get());
// destroy the guard
it->output()->replaceAllUsesWith(it->input());
it.destroyCurrent();
}
for (auto ib : it->blocks()) {
removeGuards(ib);
}
}
}
// replace each prim::Guard
// with its corresponding prim::BailOut
void replaceGuardsWithBailouts() {
for (auto e : replacements_) {
e.first->replaceAllUsesWith(e.second);
e.second->node()->insertAfter(e.first->node());
e.first->node()->destroy();
}
}
// Inserts prim::BailOut nodes for every prim::Guard
// Each BailOut point takes the set of inputs live
// at that particular execution point.
// An input is live if it's used beyond the guard/BailOut
// point to compute graph's outputs
void insertBailOuts(Block* b) {
for (auto it = b->nodes().begin(); it != b->nodes().end(); ++it) {
if (it->kind() == prim::Guard) {
auto bailout_node = b->owningGraph()->create(prim::BailOut);
bailouts_.push_back(bailout_node);
const auto& live_inputs = liveness_sets_[*it];
// guarded inputs come first
// currently, there's always one guarded input
bailout_node->addInput(it->input());
for (auto li : live_inputs) {
// Guarded inputs have already been added
// Also, skip some inputs that BailOutGraphBuilder can
// materialize into bailout graphs directly
if (!shouldBeCapturedInByBailOut(li->node()) || li == it->input()) {
continue;
}
bailout_node->addInput(li);
}
bailout_node->output()->setType(it->output()->type());
bailout_node->i_(attr::index, bailout_index_++);
// we can't immediately replace nodes since this action will corrupt
// the liveness sets of following BailOut nodes if any of their
// arguments are BailOut nodes themselves
replacements_.insert({it->output(), bailout_node->output()});
} else {
for (auto ib : it->blocks()) {
insertBailOuts(ib);
}
}
}
}
std::shared_ptr<Graph> graph_;
std::map<Node*, Node*> subgraphs;
std::size_t bailout_index_{0};
std::unordered_map<Node*, std::vector<Value*>> liveness_sets_;
std::vector<Node*> bailouts_;
std::map<Value*, Value*> replacements_;
};
void InsertBailOuts(std::shared_ptr<Graph> graph) {
BailOutInserter ibo(std::move(graph));
ibo.run();
}
// linearly scans through graph's nodes to locate prim::BailOut whose
// index matches the given `index`
static Node* locateBailOutNodeInUnoptimizedGraph(Block* b, int64_t index) {
for (auto n : b->nodes()) {
if ((n->kind() == prim::BailOut || n->kind() == prim::Guard) &&
n->hasAttribute(attr::index) && n->i(attr::index) == index) {
return n;
}
for (auto ib : n->blocks()) {
if (auto bn = locateBailOutNodeInUnoptimizedGraph(ib, index)) {
return bn;
}
}
}
return nullptr;
}
// Removes prim::BailOuts and hooks the guarded input directly
// to its users
static void removeBailouts(Block* b) {
for (auto it = b->nodes().begin(); it != b->nodes().end(); it++) {
if (it->kind() == prim::BailOut || it->kind() == prim::Guard) {
// clear profiling information
it->inputs().at(0)->setType(TensorType::get());
it->output()->replaceAllUsesWith(it->inputs().at(0));
it.destroyCurrent();
} else {
for (auto ib : it->blocks()) {
removeBailouts(ib);
}
}
}
}
// see `bailout_graph.h`
TORCH_API std::shared_ptr<Graph> BuildBailOutGraphFrom(
int64_t bailout_index,
const std::shared_ptr<Graph>& orig,
const std::shared_ptr<Graph>& target) {
auto orig_bailout_node =
locateBailOutNodeInUnoptimizedGraph(orig->block(), bailout_index);
GRAPH_DEBUG("bailout triggered for ", *orig_bailout_node);
GRAPH_DUMP("original bailout graph ", orig);
TORCH_INTERNAL_ASSERT(
orig_bailout_node->inputs().at(0)->type()->cast<FunctionType>() ==
nullptr);
TORCH_INTERNAL_ASSERT(
orig_bailout_node &&
(orig_bailout_node->kind() == prim::BailOut ||
orig_bailout_node->kind() == prim::Guard) &&
bailout_index == orig_bailout_node->i(attr::index));
BailOutGraphBuilderForNode bg(orig, target);
auto bailout_graph = bg.buildBailOutGraphFrom(orig_bailout_node);
removeBailouts(bailout_graph->block());
ClearProfilingInformation(bailout_graph);
GRAPH_DUMP("bailout_graph ", bailout_graph);
return bailout_graph;
}
} // namespace torch::jit