blob: 81c30616917a581b110fb6655a9fb286917b7167 [file] [log] [blame]
fpizlo@apple.comfc70ba62014-09-26 03:59:33 +00001/*
basile_clement@apple.comd7930292015-07-13 23:27:30 +00002 * Copyright (C) 2015 Apple Inc. All rights reserved.
fpizlo@apple.comfc70ba62014-09-26 03:59:33 +00003 *
4 * Redistribution and use in source and binary forms, with or without
5 * modification, are permitted provided that the following conditions
6 * are met:
7 * 1. Redistributions of source code must retain the above copyright
8 * notice, this list of conditions and the following disclaimer.
9 * 2. Redistributions in binary form must reproduce the above copyright
10 * notice, this list of conditions and the following disclaimer in the
11 * documentation and/or other materials provided with the distribution.
12 *
13 * THIS SOFTWARE IS PROVIDED BY APPLE INC. ``AS IS'' AND ANY
14 * EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
15 * IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR
16 * PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL APPLE INC. OR
17 * CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL,
18 * EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO,
19 * PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR
20 * PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY
21 * OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
22 * (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE
basile_clement@apple.comd7930292015-07-13 23:27:30 +000023 * OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
fpizlo@apple.comfc70ba62014-09-26 03:59:33 +000024 */
25
26#include "config.h"
27#include "DFGObjectAllocationSinkingPhase.h"
28
29#if ENABLE(DFG_JIT)
30
fpizlo@apple.comfc70ba62014-09-26 03:59:33 +000031#include "DFGBlockMapInlines.h"
fpizlo@apple.comf29186e2015-08-26 19:24:41 +000032#include "DFGClobbersExitState.h"
fpizlo@apple.com01b2d7a82015-05-13 22:14:25 +000033#include "DFGCombinedLiveness.h"
fpizlo@apple.comfc70ba62014-09-26 03:59:33 +000034#include "DFGGraph.h"
fpizlo@apple.comfc70ba62014-09-26 03:59:33 +000035#include "DFGInsertionSet.h"
basile_clement@apple.comd7930292015-07-13 23:27:30 +000036#include "DFGLazyNode.h"
fpizlo@apple.comfc70ba62014-09-26 03:59:33 +000037#include "DFGLivenessAnalysisPhase.h"
38#include "DFGOSRAvailabilityAnalysisPhase.h"
39#include "DFGPhase.h"
basile_clement@apple.comd7930292015-07-13 23:27:30 +000040#include "DFGPromotedHeapLocation.h"
fpizlo@apple.comfc70ba62014-09-26 03:59:33 +000041#include "DFGSSACalculator.h"
42#include "DFGValidate.h"
43#include "JSCInlines.h"
44
basile_clement@apple.comd7930292015-07-13 23:27:30 +000045#include <list>
46
fpizlo@apple.comfc70ba62014-09-26 03:59:33 +000047namespace JSC { namespace DFG {
48
basile_clement@apple.comd7930292015-07-13 23:27:30 +000049namespace {
50
51bool verbose = false;
52
53// In order to sink object cycles, we use a points-to analysis coupled
54// with an escape analysis. This analysis is actually similar to an
55// abstract interpreter focused on local allocations and ignoring
56// everything else.
57//
58// We represent the local heap using two mappings:
59//
60// - A set of the local allocations present in the function, where
61// each of those have a further mapping from
62// PromotedLocationDescriptor to local allocations they must point
63// to.
64//
65// - A "pointer" mapping from nodes to local allocations, if they must
66// be equal to said local allocation and are currently live. This
67// can be because the node is the actual node that created the
68// allocation, or any other node that must currently point to it -
69// we don't make a difference.
70//
71// The following graph is a motivation for why we separate allocations
72// from pointers:
73//
74// Block #0
75// 0: NewObject({})
76// 1: NewObject({})
77// -: PutByOffset(@0, @1, x)
78// -: PutStructure(@0, {x:0})
79// 2: GetByOffset(@0, x)
80// -: Jump(#1)
81//
82// Block #1
83// -: Return(@2)
84//
85// Here, we need to remember in block #1 that @2 points to a local
86// allocation with appropriate fields and structures information
87// (because we should be able to place a materialization on top of
88// block #1 here), even though @1 is dead. We *could* just keep @1
89// artificially alive here, but there is no real reason to do it:
90// after all, by the end of block #0, @1 and @2 should be completely
91// interchangeable, and there is no reason for us to artificially make
92// @1 more important.
93//
94// An important point to consider to understand this separation is
95// that we should think of the local heap as follow: we have a
96// bunch of nodes that are pointers to "allocations" that live
97// someplace on the heap, and those allocations can have pointers in
98// between themselves as well. We shouldn't care about whatever
99// names we give to the allocations ; what matters when
100// comparing/merging two heaps is the isomorphism/comparison between
101// the allocation graphs as seen by the nodes.
102//
103// For instance, in the following graph:
104//
105// Block #0
106// 0: NewObject({})
107// -: Branch(#1, #2)
108//
109// Block #1
110// 1: NewObject({})
111// -: PutByOffset(@0, @1, x)
112// -: PutStructure(@0, {x:0})
113// -: Jump(#3)
114//
115// Block #2
116// 2: NewObject({})
117// -: PutByOffset(@2, undefined, x)
118// -: PutStructure(@2, {x:0})
119// -: PutByOffset(@0, @2, x)
120// -: PutStructure(@0, {x:0})
121// -: Jump(#3)
122//
123// Block #3
124// -: Return(@0)
125//
126// we should think of the heaps at tail of blocks #1 and #2 as being
127// exactly the same, even though one has @0.x pointing to @1 and the
128// other has @0.x pointing to @2, because in essence this should not
129// be different from the graph where we hoisted @1 and @2 into a
130// single allocation in block #0. We currently will not handle this
131// case, because we merge allocations based on the node they are
132// coming from, but this is only a technicality for the sake of
133// simplicity that shouldn't hide the deeper idea outlined here.
134
135class Allocation {
136public:
137 // We use Escaped as a special allocation kind because when we
138 // decide to sink an allocation, we still need to keep track of it
139 // once it is escaped if it still has pointers to it in order to
140 // replace any use of those pointers by the corresponding
141 // materialization
gskachkov@gmail.com42c26422016-04-11 19:04:33 +0000142 enum class Kind { Escaped, Object, Activation, Function, GeneratorFunction };
basile_clement@apple.comd7930292015-07-13 23:27:30 +0000143
144 explicit Allocation(Node* identifier = nullptr, Kind kind = Kind::Escaped)
145 : m_identifier(identifier)
146 , m_kind(kind)
147 {
148 }
149
150
151 const HashMap<PromotedLocationDescriptor, Node*>& fields() const
152 {
153 return m_fields;
154 }
155
156 Node* get(PromotedLocationDescriptor descriptor)
157 {
158 return m_fields.get(descriptor);
159 }
160
161 Allocation& set(PromotedLocationDescriptor descriptor, Node* value)
162 {
163 // Pointing to anything else than an unescaped local
164 // allocation is represented by simply not having the
165 // field
166 if (value)
167 m_fields.set(descriptor, value);
168 else
169 m_fields.remove(descriptor);
170 return *this;
171 }
172
173 void remove(PromotedLocationDescriptor descriptor)
174 {
175 set(descriptor, nullptr);
176 }
177
178 bool hasStructures() const
179 {
180 switch (kind()) {
181 case Kind::Object:
182 return true;
183
184 default:
185 return false;
186 }
187 }
188
189 Allocation& setStructures(const StructureSet& structures)
190 {
191 ASSERT(hasStructures() && !structures.isEmpty());
192 m_structures = structures;
193 return *this;
194 }
195
196 Allocation& mergeStructures(const StructureSet& structures)
197 {
198 ASSERT(hasStructures() || structures.isEmpty());
199 m_structures.merge(structures);
200 return *this;
201 }
202
203 Allocation& filterStructures(const StructureSet& structures)
204 {
205 ASSERT(hasStructures());
206 m_structures.filter(structures);
207 return *this;
208 }
209
210 const StructureSet& structures() const
211 {
212 return m_structures;
213 }
214
215 Node* identifier() const { return m_identifier; }
216
217 Kind kind() const { return m_kind; }
218
219 bool isEscapedAllocation() const
220 {
221 return kind() == Kind::Escaped;
222 }
223
224 bool isObjectAllocation() const
225 {
226 return m_kind == Kind::Object;
227 }
228
229 bool isActivationAllocation() const
230 {
231 return m_kind == Kind::Activation;
232 }
233
234 bool isFunctionAllocation() const
235 {
gskachkov@gmail.com42c26422016-04-11 19:04:33 +0000236 return m_kind == Kind::Function || m_kind == Kind::GeneratorFunction;
basile_clement@apple.comd7930292015-07-13 23:27:30 +0000237 }
238
239 bool operator==(const Allocation& other) const
240 {
241 return m_identifier == other.m_identifier
242 && m_kind == other.m_kind
243 && m_fields == other.m_fields
244 && m_structures == other.m_structures;
245 }
246
247 bool operator!=(const Allocation& other) const
248 {
249 return !(*this == other);
250 }
251
252 void dump(PrintStream& out) const
253 {
254 dumpInContext(out, nullptr);
255 }
256
257 void dumpInContext(PrintStream& out, DumpContext* context) const
258 {
259 switch (m_kind) {
260 case Kind::Escaped:
261 out.print("Escaped");
262 break;
263
264 case Kind::Object:
265 out.print("Object");
266 break;
267
268 case Kind::Function:
269 out.print("Function");
270 break;
utatane.tea@gmail.com9d9fd322015-12-17 10:33:08 +0000271
272 case Kind::GeneratorFunction:
273 out.print("GeneratorFunction");
commit-queue@webkit.orga4201b02015-08-17 22:24:20 +0000274 break;
basile_clement@apple.comd7930292015-07-13 23:27:30 +0000275
276 case Kind::Activation:
277 out.print("Activation");
278 break;
279 }
280 out.print("Allocation(");
281 if (!m_structures.isEmpty())
282 out.print(inContext(m_structures, context));
283 if (!m_fields.isEmpty()) {
284 if (!m_structures.isEmpty())
285 out.print(", ");
286 out.print(mapDump(m_fields, " => #", ", "));
287 }
288 out.print(")");
289 }
290
291private:
292 Node* m_identifier; // This is the actual node that created the allocation
293 Kind m_kind;
294 HashMap<PromotedLocationDescriptor, Node*> m_fields;
295 StructureSet m_structures;
296};
297
298class LocalHeap {
299public:
300 Allocation& newAllocation(Node* node, Allocation::Kind kind)
301 {
302 ASSERT(!m_pointers.contains(node) && !isAllocation(node));
303 m_pointers.add(node, node);
304 return m_allocations.set(node, Allocation(node, kind)).iterator->value;
305 }
306
307 bool isAllocation(Node* identifier) const
308 {
309 return m_allocations.contains(identifier);
310 }
311
312 // Note that this is fundamentally different from
313 // onlyLocalAllocation() below. getAllocation() takes as argument
314 // a node-as-identifier, that is, an allocation node. This
315 // allocation node doesn't have to be alive; it may only be
316 // pointed to by other nodes or allocation fields.
317 // For instance, in the following graph:
318 //
319 // Block #0
320 // 0: NewObject({})
321 // 1: NewObject({})
322 // -: PutByOffset(@0, @1, x)
323 // -: PutStructure(@0, {x:0})
324 // 2: GetByOffset(@0, x)
325 // -: Jump(#1)
326 //
327 // Block #1
328 // -: Return(@2)
329 //
330 // At head of block #1, the only reachable allocation is #@1,
331 // which can be reached through node @2. Thus, getAllocation(#@1)
332 // contains the appropriate metadata for this allocation, but
333 // onlyLocalAllocation(@1) is null, as @1 is no longer a pointer
334 // to #@1 (since it is dead). Conversely, onlyLocalAllocation(@2)
335 // is the same as getAllocation(#@1), while getAllocation(#@2)
336 // does not make sense since @2 is not an allocation node.
337 //
338 // This is meant to be used when the node is already known to be
339 // an identifier (i.e. an allocation) - probably because it was
340 // found as value of a field or pointer in the current heap, or
341 // was the result of a call to follow(). In any other cases (such
342 // as when doing anything while traversing the graph), the
343 // appropriate function to call is probably onlyLocalAllocation.
344 Allocation& getAllocation(Node* identifier)
345 {
346 auto iter = m_allocations.find(identifier);
347 ASSERT(iter != m_allocations.end());
348 return iter->value;
349 }
350
351 void newPointer(Node* node, Node* identifier)
352 {
353 ASSERT(!m_allocations.contains(node) && !m_pointers.contains(node));
354 ASSERT(isAllocation(identifier));
355 m_pointers.add(node, identifier);
356 }
357
358 // follow solves the points-to problem. Given a live node, which
359 // may be either an allocation itself or a heap read (e.g. a
360 // GetByOffset node), it returns the corresponding allocation
361 // node, if there is one. If the argument node is neither an
362 // allocation or a heap read, or may point to different nodes,
363 // nullptr will be returned. Note that a node that points to
364 // different nodes can never point to an unescaped local
365 // allocation.
366 Node* follow(Node* node) const
367 {
368 auto iter = m_pointers.find(node);
369 ASSERT(iter == m_pointers.end() || m_allocations.contains(iter->value));
370 return iter == m_pointers.end() ? nullptr : iter->value;
371 }
372
373 Node* follow(PromotedHeapLocation location) const
374 {
375 const Allocation& base = m_allocations.find(location.base())->value;
376 auto iter = base.fields().find(location.descriptor());
377
378 if (iter == base.fields().end())
379 return nullptr;
380
381 return iter->value;
382 }
383
384 // onlyLocalAllocation find the corresponding allocation metadata
385 // for any live node. onlyLocalAllocation(node) is essentially
386 // getAllocation(follow(node)), with appropriate null handling.
387 Allocation* onlyLocalAllocation(Node* node)
388 {
389 Node* identifier = follow(node);
390 if (!identifier)
391 return nullptr;
392
393 return &getAllocation(identifier);
394 }
395
396 Allocation* onlyLocalAllocation(PromotedHeapLocation location)
397 {
398 Node* identifier = follow(location);
399 if (!identifier)
400 return nullptr;
401
402 return &getAllocation(identifier);
403 }
404
405 // This allows us to store the escapees only when necessary. If
406 // set, the current escapees can be retrieved at any time using
407 // takeEscapees(), which will clear the cached set of escapees;
408 // otherwise the heap won't remember escaping allocations.
409 void setWantEscapees()
410 {
411 m_wantEscapees = true;
412 }
413
414 HashMap<Node*, Allocation> takeEscapees()
415 {
aestes@apple.com13aae082016-01-02 08:03:08 +0000416 return WTFMove(m_escapees);
basile_clement@apple.comd7930292015-07-13 23:27:30 +0000417 }
418
419 void escape(Node* node)
420 {
421 Node* identifier = follow(node);
422 if (!identifier)
423 return;
424
425 escapeAllocation(identifier);
426 }
427
428 void merge(const LocalHeap& other)
429 {
430 assertIsValid();
431 other.assertIsValid();
432 ASSERT(!m_wantEscapees);
433
434 if (!reached()) {
435 ASSERT(other.reached());
436 *this = other;
437 return;
438 }
439
440 HashSet<Node*> toEscape;
441
442 for (auto& allocationEntry : other.m_allocations)
443 m_allocations.add(allocationEntry.key, allocationEntry.value);
444 for (auto& allocationEntry : m_allocations) {
445 auto allocationIter = other.m_allocations.find(allocationEntry.key);
446
447 // If we have it and they don't, it died for them but we
448 // are keeping it alive from another field somewhere.
449 // There is nothing to do - we will be escaped
450 // automatically when we handle that other field.
451 // This will also happen for allocation that we have and
452 // they don't, and all of those will get pruned.
453 if (allocationIter == other.m_allocations.end())
454 continue;
455
456 if (allocationEntry.value.kind() != allocationIter->value.kind()) {
457 toEscape.add(allocationEntry.key);
458 for (const auto& fieldEntry : allocationIter->value.fields())
459 toEscape.add(fieldEntry.value);
460 } else {
461 mergePointerSets(
462 allocationEntry.value.fields(), allocationIter->value.fields(),
463 [&] (Node* identifier) {
464 toEscape.add(identifier);
465 },
466 [&] (PromotedLocationDescriptor field) {
467 allocationEntry.value.remove(field);
468 });
469 allocationEntry.value.mergeStructures(allocationIter->value.structures());
470 }
471 }
472
473 mergePointerSets(m_pointers, other.m_pointers,
474 [&] (Node* identifier) {
475 toEscape.add(identifier);
476 },
477 [&] (Node* field) {
478 m_pointers.remove(field);
479 });
480
481 for (Node* identifier : toEscape)
482 escapeAllocation(identifier);
483
484 if (!ASSERT_DISABLED) {
485 for (const auto& entry : m_allocations)
486 ASSERT_UNUSED(entry, entry.value.isEscapedAllocation() || other.m_allocations.contains(entry.key));
487 }
488
489 // If there is no remaining pointer to an allocation, we can
490 // remove it. This should only happen for escaped allocations,
491 // because we only merge liveness-pruned heaps in the first
492 // place.
493 prune();
494
495 assertIsValid();
496 }
497
498 void pruneByLiveness(const HashSet<Node*>& live)
499 {
500 Vector<Node*> toRemove;
501 for (const auto& entry : m_pointers) {
502 if (!live.contains(entry.key))
503 toRemove.append(entry.key);
504 }
505 for (Node* node : toRemove)
506 m_pointers.remove(node);
507
508 prune();
509 }
510
511 void assertIsValid() const
512 {
513 if (ASSERT_DISABLED)
514 return;
515
516 // Pointers should point to an actual allocation
517 for (const auto& entry : m_pointers) {
518 ASSERT_UNUSED(entry, entry.value);
519 ASSERT(m_allocations.contains(entry.value));
520 }
521
522 for (const auto& allocationEntry : m_allocations) {
523 // Fields should point to an actual allocation
524 for (const auto& fieldEntry : allocationEntry.value.fields()) {
525 ASSERT_UNUSED(fieldEntry, fieldEntry.value);
526 ASSERT(m_allocations.contains(fieldEntry.value));
527 }
528 }
529 }
530
531 bool operator==(const LocalHeap& other) const
532 {
533 assertIsValid();
534 other.assertIsValid();
535 return m_allocations == other.m_allocations
536 && m_pointers == other.m_pointers;
537 }
538
539 bool operator!=(const LocalHeap& other) const
540 {
541 return !(*this == other);
542 }
543
basile_clement@apple.com17e04972015-07-21 20:15:54 +0000544 const HashMap<Node*, Allocation>& allocations() const
basile_clement@apple.comd7930292015-07-13 23:27:30 +0000545 {
546 return m_allocations;
547 }
548
basile_clement@apple.com17e04972015-07-21 20:15:54 +0000549 const HashMap<Node*, Node*>& pointers() const
basile_clement@apple.comd7930292015-07-13 23:27:30 +0000550 {
551 return m_pointers;
552 }
553
554 void dump(PrintStream& out) const
555 {
556 out.print(" Allocations:\n");
557 for (const auto& entry : m_allocations)
558 out.print(" #", entry.key, ": ", entry.value, "\n");
559 out.print(" Pointers:\n");
560 for (const auto& entry : m_pointers)
561 out.print(" ", entry.key, " => #", entry.value, "\n");
562 }
563
564 bool reached() const
565 {
566 return m_reached;
567 }
568
569 void setReached()
570 {
571 m_reached = true;
572 }
573
574private:
575 // When we merge two heaps, we escape all fields of allocations,
576 // unless they point to the same thing in both heaps.
577 // The reason for this is that it allows us not to do extra work
578 // for diamond graphs where we would otherwise have to check
579 // whether we have a single definition or not, which would be
580 // cumbersome.
581 //
582 // Note that we should try to unify nodes even when they are not
583 // from the same allocation; for instance we should be able to
584 // completely eliminate all allocations from the following graph:
585 //
586 // Block #0
587 // 0: NewObject({})
588 // -: Branch(#1, #2)
589 //
590 // Block #1
591 // 1: NewObject({})
592 // -: PutByOffset(@1, "left", val)
593 // -: PutStructure(@1, {val:0})
594 // -: PutByOffset(@0, @1, x)
595 // -: PutStructure(@0, {x:0})
596 // -: Jump(#3)
597 //
598 // Block #2
599 // 2: NewObject({})
600 // -: PutByOffset(@2, "right", val)
601 // -: PutStructure(@2, {val:0})
602 // -: PutByOffset(@0, @2, x)
603 // -: PutStructure(@0, {x:0})
604 // -: Jump(#3)
605 //
606 // Block #3:
607 // 3: GetByOffset(@0, x)
608 // 4: GetByOffset(@3, val)
609 // -: Return(@4)
610 template<typename Key, typename EscapeFunctor, typename RemoveFunctor>
611 void mergePointerSets(
612 const HashMap<Key, Node*>& my, const HashMap<Key, Node*>& their,
613 const EscapeFunctor& escape, const RemoveFunctor& remove)
614 {
615 Vector<Key> toRemove;
616 for (const auto& entry : my) {
617 auto iter = their.find(entry.key);
618 if (iter == their.end()) {
619 toRemove.append(entry.key);
620 escape(entry.value);
621 } else if (iter->value != entry.value) {
622 toRemove.append(entry.key);
623 escape(entry.value);
624 escape(iter->value);
625 }
626 }
627 for (const auto& entry : their) {
628 if (my.contains(entry.key))
629 continue;
630 escape(entry.value);
631 }
632 for (Key key : toRemove)
633 remove(key);
634 }
635
636 void escapeAllocation(Node* identifier)
637 {
638 Allocation& allocation = getAllocation(identifier);
639 if (allocation.isEscapedAllocation())
640 return;
641
aestes@apple.com13aae082016-01-02 08:03:08 +0000642 Allocation unescaped = WTFMove(allocation);
basile_clement@apple.comd7930292015-07-13 23:27:30 +0000643 allocation = Allocation(unescaped.identifier(), Allocation::Kind::Escaped);
644
645 for (const auto& entry : unescaped.fields())
646 escapeAllocation(entry.value);
647
648 if (m_wantEscapees)
aestes@apple.com13aae082016-01-02 08:03:08 +0000649 m_escapees.add(unescaped.identifier(), WTFMove(unescaped));
basile_clement@apple.comd7930292015-07-13 23:27:30 +0000650 }
651
652 void prune()
653 {
654 HashSet<Node*> reachable;
655 for (const auto& entry : m_pointers)
656 reachable.add(entry.value);
657
658 // Repeatedly mark as reachable allocations in fields of other
659 // reachable allocations
660 {
661 Vector<Node*> worklist;
662 worklist.appendRange(reachable.begin(), reachable.end());
663
664 while (!worklist.isEmpty()) {
665 Node* identifier = worklist.takeLast();
666 Allocation& allocation = m_allocations.find(identifier)->value;
667 for (const auto& entry : allocation.fields()) {
668 if (reachable.add(entry.value).isNewEntry)
669 worklist.append(entry.value);
670 }
671 }
672 }
673
674 // Remove unreachable allocations
675 {
676 Vector<Node*> toRemove;
677 for (const auto& entry : m_allocations) {
678 if (!reachable.contains(entry.key))
679 toRemove.append(entry.key);
680 }
681 for (Node* identifier : toRemove)
682 m_allocations.remove(identifier);
683 }
684 }
685
686 bool m_reached = false;
687 HashMap<Node*, Node*> m_pointers;
688 HashMap<Node*, Allocation> m_allocations;
689
690 bool m_wantEscapees = false;
691 HashMap<Node*, Allocation> m_escapees;
692};
fpizlo@apple.comfc70ba62014-09-26 03:59:33 +0000693
694class ObjectAllocationSinkingPhase : public Phase {
695public:
696 ObjectAllocationSinkingPhase(Graph& graph)
basile_clement@apple.comd7930292015-07-13 23:27:30 +0000697 : Phase(graph, "object allocation elimination")
698 , m_pointerSSA(graph)
699 , m_allocationSSA(graph)
fpizlo@apple.comfc70ba62014-09-26 03:59:33 +0000700 , m_insertionSet(graph)
701 {
702 }
basile_clement@apple.comd7930292015-07-13 23:27:30 +0000703
fpizlo@apple.comfc70ba62014-09-26 03:59:33 +0000704 bool run()
705 {
basile_clement@apple.comd7930292015-07-13 23:27:30 +0000706 ASSERT(m_graph.m_form == SSA);
fpizlo@apple.comfc70ba62014-09-26 03:59:33 +0000707 ASSERT(m_graph.m_fixpointState == FixpointNotConverged);
basile_clement@apple.comd7930292015-07-13 23:27:30 +0000708
fpizlo@apple.comfc70ba62014-09-26 03:59:33 +0000709 if (!performSinking())
710 return false;
basile_clement@apple.comd7930292015-07-13 23:27:30 +0000711
fpizlo@apple.comfc70ba62014-09-26 03:59:33 +0000712 if (verbose) {
basile_clement@apple.comd7930292015-07-13 23:27:30 +0000713 dataLog("Graph after elimination:\n");
fpizlo@apple.comfc70ba62014-09-26 03:59:33 +0000714 m_graph.dump();
715 }
basile_clement@apple.comd7930292015-07-13 23:27:30 +0000716
fpizlo@apple.comfc70ba62014-09-26 03:59:33 +0000717 return true;
718 }
719
720private:
721 bool performSinking()
722 {
723 m_graph.computeRefCounts();
basile_clement@apple.comd7930292015-07-13 23:27:30 +0000724 m_graph.initializeNodeOwners();
fpizlo@apple.com92dbc1f2015-11-02 00:48:03 +0000725 m_graph.ensureDominators();
fpizlo@apple.comfc70ba62014-09-26 03:59:33 +0000726 performLivenessAnalysis(m_graph);
727 performOSRAvailabilityAnalysis(m_graph);
fpizlo@apple.com01b2d7a82015-05-13 22:14:25 +0000728 m_combinedLiveness = CombinedLiveness(m_graph);
basile_clement@apple.comd7930292015-07-13 23:27:30 +0000729
fpizlo@apple.comfc70ba62014-09-26 03:59:33 +0000730 CString graphBeforeSinking;
731 if (Options::verboseValidationFailure() && Options::validateGraphAtEachPhase()) {
732 StringPrintStream out;
733 m_graph.dump(out);
734 graphBeforeSinking = out.toCString();
735 }
basile_clement@apple.comd7930292015-07-13 23:27:30 +0000736
fpizlo@apple.comfc70ba62014-09-26 03:59:33 +0000737 if (verbose) {
basile_clement@apple.comd7930292015-07-13 23:27:30 +0000738 dataLog("Graph before elimination:\n");
fpizlo@apple.comfc70ba62014-09-26 03:59:33 +0000739 m_graph.dump();
740 }
basile_clement@apple.comd7930292015-07-13 23:27:30 +0000741
742 performAnalysis();
743
744 if (!determineSinkCandidates())
fpizlo@apple.comfc70ba62014-09-26 03:59:33 +0000745 return false;
basile_clement@apple.comd7930292015-07-13 23:27:30 +0000746
747 if (verbose) {
748 for (BasicBlock* block : m_graph.blocksInNaturalOrder()) {
749 dataLog("Heap at head of ", *block, ": \n", m_heapAtHead[block]);
750 dataLog("Heap at tail of ", *block, ": \n", m_heapAtTail[block]);
751 }
752 }
753
754 promoteLocalHeap();
755
fpizlo@apple.comfc70ba62014-09-26 03:59:33 +0000756 if (Options::validateGraphAtEachPhase())
fpizlo@apple.comf29186e2015-08-26 19:24:41 +0000757 DFG::validate(m_graph, DumpGraph, graphBeforeSinking);
fpizlo@apple.comfc70ba62014-09-26 03:59:33 +0000758 return true;
759 }
basile_clement@apple.comd7930292015-07-13 23:27:30 +0000760
761 void performAnalysis()
fpizlo@apple.comfc70ba62014-09-26 03:59:33 +0000762 {
basile_clement@apple.comd7930292015-07-13 23:27:30 +0000763 m_heapAtHead = BlockMap<LocalHeap>(m_graph);
764 m_heapAtTail = BlockMap<LocalHeap>(m_graph);
765
fpizlo@apple.comfc70ba62014-09-26 03:59:33 +0000766 bool changed;
767 do {
768 if (verbose)
basile_clement@apple.comd7930292015-07-13 23:27:30 +0000769 dataLog("Doing iteration of escape analysis.\n");
fpizlo@apple.comfc70ba62014-09-26 03:59:33 +0000770 changed = false;
basile_clement@apple.comd7930292015-07-13 23:27:30 +0000771
772 for (BasicBlock* block : m_graph.blocksInPreOrder()) {
773 m_heapAtHead[block].setReached();
774 m_heap = m_heapAtHead[block];
775
fpizlo@apple.comfc70ba62014-09-26 03:59:33 +0000776 for (Node* node : *block) {
777 handleNode(
778 node,
basile_clement@apple.comd7930292015-07-13 23:27:30 +0000779 [] (PromotedHeapLocation, LazyNode) { },
780 [&] (PromotedHeapLocation) -> Node* {
781 return nullptr;
fpizlo@apple.comfc70ba62014-09-26 03:59:33 +0000782 });
783 }
basile_clement@apple.comd7930292015-07-13 23:27:30 +0000784
785 if (m_heap == m_heapAtTail[block])
fpizlo@apple.comfc70ba62014-09-26 03:59:33 +0000786 continue;
basile_clement@apple.comd7930292015-07-13 23:27:30 +0000787
788 m_heapAtTail[block] = m_heap;
fpizlo@apple.comfc70ba62014-09-26 03:59:33 +0000789 changed = true;
basile_clement@apple.comd7930292015-07-13 23:27:30 +0000790
791 m_heap.assertIsValid();
792
793 // We keep only pointers that are live, and only
794 // allocations that are either live, pointed to by a
795 // live pointer, or (recursively) stored in a field of
796 // a live allocation.
797 //
798 // This means we can accidentaly leak non-dominating
799 // nodes into the successor. However, due to the
800 // non-dominance property, we are guaranteed that the
801 // successor has at least one predecessor that is not
802 // dominated either: this means any reference to a
803 // non-dominating allocation in the successor will
804 // trigger an escape and get pruned during the merge.
805 m_heap.pruneByLiveness(m_combinedLiveness.liveAtTail[block]);
806
807 for (BasicBlock* successorBlock : block->successors())
808 m_heapAtHead[successorBlock].merge(m_heap);
fpizlo@apple.comfc70ba62014-09-26 03:59:33 +0000809 }
810 } while (changed);
fpizlo@apple.comfc70ba62014-09-26 03:59:33 +0000811 }
812
basile_clement@apple.comd7930292015-07-13 23:27:30 +0000813 template<typename WriteFunctor, typename ResolveFunctor>
fpizlo@apple.comfc70ba62014-09-26 03:59:33 +0000814 void handleNode(
815 Node* node,
basile_clement@apple.comd7930292015-07-13 23:27:30 +0000816 const WriteFunctor& heapWrite,
817 const ResolveFunctor& heapResolve)
fpizlo@apple.comfc70ba62014-09-26 03:59:33 +0000818 {
basile_clement@apple.comd7930292015-07-13 23:27:30 +0000819 m_heap.assertIsValid();
820 ASSERT(m_heap.takeEscapees().isEmpty());
821
822 Allocation* target = nullptr;
823 HashMap<PromotedLocationDescriptor, LazyNode> writes;
824 PromotedLocationDescriptor exactRead;
825
fpizlo@apple.comfc70ba62014-09-26 03:59:33 +0000826 switch (node->op()) {
827 case NewObject:
basile_clement@apple.comd7930292015-07-13 23:27:30 +0000828 target = &m_heap.newAllocation(node, Allocation::Kind::Object);
829 target->setStructures(node->structure());
830 writes.add(
831 StructurePLoc, LazyNode(m_graph.freeze(node->structure())));
fpizlo@apple.comfc70ba62014-09-26 03:59:33 +0000832 break;
commit-queue@webkit.org88ab4e72015-04-24 02:23:36 +0000833
commit-queue@webkit.orga4201b02015-08-17 22:24:20 +0000834 case NewFunction:
utatane.tea@gmail.com9d9fd322015-12-17 10:33:08 +0000835 case NewGeneratorFunction: {
fpizlo@apple.combe625d52016-03-16 20:12:27 +0000836 if (isStillValid(node->castOperand<FunctionExecutable*>()->singletonFunction())) {
basile_clement@apple.comd7930292015-07-13 23:27:30 +0000837 m_heap.escape(node->child1().node());
838 break;
839 }
commit-queue@webkit.orga4201b02015-08-17 22:24:20 +0000840
utatane.tea@gmail.com9d9fd322015-12-17 10:33:08 +0000841 if (node->op() == NewGeneratorFunction)
842 target = &m_heap.newAllocation(node, Allocation::Kind::GeneratorFunction);
utatane.tea@gmail.com9d9fd322015-12-17 10:33:08 +0000843 else
844 target = &m_heap.newAllocation(node, Allocation::Kind::Function);
basile_clement@apple.comd7930292015-07-13 23:27:30 +0000845 writes.add(FunctionExecutablePLoc, LazyNode(node->cellOperand()));
846 writes.add(FunctionActivationPLoc, LazyNode(node->child1().node()));
847 break;
848 }
849
850 case CreateActivation: {
fpizlo@apple.combe625d52016-03-16 20:12:27 +0000851 if (isStillValid(node->castOperand<SymbolTable*>()->singletonScope())) {
basile_clement@apple.comd7930292015-07-13 23:27:30 +0000852 m_heap.escape(node->child1().node());
853 break;
854 }
855 target = &m_heap.newAllocation(node, Allocation::Kind::Activation);
856 writes.add(ActivationSymbolTablePLoc, LazyNode(node->cellOperand()));
857 writes.add(ActivationScopePLoc, LazyNode(node->child1().node()));
858 {
859 SymbolTable* symbolTable = node->castOperand<SymbolTable*>();
860 ConcurrentJITLocker locker(symbolTable->m_lock);
saambarati1@gmail.com144f17c2015-07-15 21:41:08 +0000861 LazyNode initialValue(m_graph.freeze(node->initializationValueForActivation()));
basile_clement@apple.comd7930292015-07-13 23:27:30 +0000862 for (auto iter = symbolTable->begin(locker), end = symbolTable->end(locker); iter != end; ++iter) {
863 writes.add(
864 PromotedLocationDescriptor(ClosureVarPLoc, iter->value.scopeOffset().offset()),
saambarati1@gmail.com144f17c2015-07-15 21:41:08 +0000865 initialValue);
basile_clement@apple.comd7930292015-07-13 23:27:30 +0000866 }
867 }
868 break;
869 }
870
basile_clement@apple.comd7930292015-07-13 23:27:30 +0000871 case PutStructure:
872 target = m_heap.onlyLocalAllocation(node->child1().node());
873 if (target && target->isObjectAllocation()) {
874 writes.add(StructurePLoc, LazyNode(m_graph.freeze(JSValue(node->transition()->next))));
875 target->setStructures(node->transition()->next);
876 } else
877 m_heap.escape(node->child1().node());
commit-queue@webkit.org88ab4e72015-04-24 02:23:36 +0000878 break;
879
basile_clement@apple.comd7930292015-07-13 23:27:30 +0000880 case CheckStructure: {
881 Allocation* allocation = m_heap.onlyLocalAllocation(node->child1().node());
882 if (allocation && allocation->isObjectAllocation()) {
883 allocation->filterStructures(node->structureSet());
884 if (Node* value = heapResolve(PromotedHeapLocation(allocation->identifier(), StructurePLoc)))
885 node->convertToCheckStructureImmediate(value);
886 } else
887 m_heap.escape(node->child1().node());
888 break;
889 }
890
891 case GetByOffset:
892 case GetGetterSetterByOffset:
893 target = m_heap.onlyLocalAllocation(node->child2().node());
894 if (target && target->isObjectAllocation()) {
895 unsigned identifierNumber = node->storageAccessData().identifierNumber;
896 exactRead = PromotedLocationDescriptor(NamedPropertyPLoc, identifierNumber);
897 } else {
898 m_heap.escape(node->child1().node());
899 m_heap.escape(node->child2().node());
900 }
901 break;
902
basile_clement@apple.com50af4b82015-08-15 05:00:57 +0000903 case MultiGetByOffset: {
904 Allocation* allocation = m_heap.onlyLocalAllocation(node->child1().node());
905 if (allocation && allocation->isObjectAllocation()) {
906 MultiGetByOffsetData& data = node->multiGetByOffsetData();
907 StructureSet validStructures;
908 bool hasInvalidStructures = false;
909 for (const auto& multiGetByOffsetCase : data.cases) {
910 if (!allocation->structures().overlaps(multiGetByOffsetCase.set()))
911 continue;
912
913 switch (multiGetByOffsetCase.method().kind()) {
914 case GetByOffsetMethod::LoadFromPrototype: // We need to escape those
915 case GetByOffsetMethod::Constant: // We don't really have a way of expressing this
916 hasInvalidStructures = true;
917 break;
918
919 case GetByOffsetMethod::Load: // We're good
920 validStructures.merge(multiGetByOffsetCase.set());
921 break;
922
923 default:
924 RELEASE_ASSERT_NOT_REACHED();
925 }
926 }
927 if (hasInvalidStructures) {
928 m_heap.escape(node->child1().node());
929 break;
930 }
931 unsigned identifierNumber = data.identifierNumber;
932 PromotedHeapLocation location(NamedPropertyPLoc, allocation->identifier(), identifierNumber);
933 if (Node* value = heapResolve(location)) {
934 if (allocation->structures().isSubsetOf(validStructures))
935 node->replaceWith(value);
936 else {
937 Node* structure = heapResolve(PromotedHeapLocation(allocation->identifier(), StructurePLoc));
938 ASSERT(structure);
939 allocation->filterStructures(validStructures);
940 node->convertToCheckStructure(m_graph.addStructureSet(allocation->structures()));
941 node->convertToCheckStructureImmediate(structure);
942 node->setReplacement(value);
943 }
944 } else if (!allocation->structures().isSubsetOf(validStructures)) {
945 // Even though we don't need the result here, we still need
946 // to make the call to tell our caller that we could need
947 // the StructurePLoc.
948 // The reason for this is that when we decide not to sink a
949 // node, we will still lower any read to its fields before
950 // it escapes (which are usually reads across a function
951 // call that DFGClobberize can't handle) - but we only do
952 // this for PromotedHeapLocations that we have seen read
953 // during the analysis!
954 heapResolve(PromotedHeapLocation(allocation->identifier(), StructurePLoc));
955 allocation->filterStructures(validStructures);
956 }
957 Node* identifier = allocation->get(location.descriptor());
958 if (identifier)
959 m_heap.newPointer(node, identifier);
basile_clement@apple.comd7930292015-07-13 23:27:30 +0000960 } else
961 m_heap.escape(node->child1().node());
962 break;
basile_clement@apple.com50af4b82015-08-15 05:00:57 +0000963 }
basile_clement@apple.comd7930292015-07-13 23:27:30 +0000964
965 case PutByOffset:
966 target = m_heap.onlyLocalAllocation(node->child2().node());
967 if (target && target->isObjectAllocation()) {
968 unsigned identifierNumber = node->storageAccessData().identifierNumber;
969 writes.add(
970 PromotedLocationDescriptor(NamedPropertyPLoc, identifierNumber),
971 LazyNode(node->child3().node()));
972 } else {
973 m_heap.escape(node->child1().node());
974 m_heap.escape(node->child2().node());
975 m_heap.escape(node->child3().node());
976 }
977 break;
978
979 case GetClosureVar:
980 target = m_heap.onlyLocalAllocation(node->child1().node());
981 if (target && target->isActivationAllocation()) {
982 exactRead =
983 PromotedLocationDescriptor(ClosureVarPLoc, node->scopeOffset().offset());
984 } else
985 m_heap.escape(node->child1().node());
986 break;
987
988 case PutClosureVar:
989 target = m_heap.onlyLocalAllocation(node->child1().node());
990 if (target && target->isActivationAllocation()) {
991 writes.add(
992 PromotedLocationDescriptor(ClosureVarPLoc, node->scopeOffset().offset()),
993 LazyNode(node->child2().node()));
994 } else {
995 m_heap.escape(node->child1().node());
996 m_heap.escape(node->child2().node());
997 }
998 break;
999
1000 case SkipScope:
1001 target = m_heap.onlyLocalAllocation(node->child1().node());
1002 if (target && target->isActivationAllocation())
1003 exactRead = ActivationScopePLoc;
1004 else
1005 m_heap.escape(node->child1().node());
1006 break;
1007
1008 case GetExecutable:
1009 target = m_heap.onlyLocalAllocation(node->child1().node());
1010 if (target && target->isFunctionAllocation())
1011 exactRead = FunctionExecutablePLoc;
1012 else
1013 m_heap.escape(node->child1().node());
1014 break;
1015
1016 case GetScope:
1017 target = m_heap.onlyLocalAllocation(node->child1().node());
1018 if (target && target->isFunctionAllocation())
1019 exactRead = FunctionActivationPLoc;
1020 else
1021 m_heap.escape(node->child1().node());
fpizlo@apple.com09b14f92015-05-13 05:21:16 +00001022 break;
1023
1024 case Check:
basile_clement@apple.com2ca1f7b2015-05-05 16:34:21 +00001025 m_graph.doToChildren(
1026 node,
1027 [&] (Edge edge) {
fpizlo@apple.com07e90782015-05-13 17:39:02 +00001028 if (edge.willNotHaveCheck())
1029 return;
basile_clement@apple.comd7930292015-07-13 23:27:30 +00001030
fpizlo@apple.com07e90782015-05-13 17:39:02 +00001031 if (alreadyChecked(edge.useKind(), SpecObject))
1032 return;
basile_clement@apple.comd7930292015-07-13 23:27:30 +00001033
1034 m_heap.escape(edge.node());
basile_clement@apple.com2ca1f7b2015-05-05 16:34:21 +00001035 });
1036 break;
1037
fpizlo@apple.comfc70ba62014-09-26 03:59:33 +00001038 case MovHint:
fpizlo@apple.com8ff74712015-03-17 15:50:44 +00001039 case PutHint:
basile_clement@apple.comd7930292015-07-13 23:27:30 +00001040 // Handled by OSR availability analysis
fpizlo@apple.comfc70ba62014-09-26 03:59:33 +00001041 break;
1042
1043 default:
1044 m_graph.doToChildren(
1045 node,
1046 [&] (Edge edge) {
basile_clement@apple.comd7930292015-07-13 23:27:30 +00001047 m_heap.escape(edge.node());
fpizlo@apple.comfc70ba62014-09-26 03:59:33 +00001048 });
1049 break;
1050 }
basile_clement@apple.comd7930292015-07-13 23:27:30 +00001051
1052 if (exactRead) {
1053 ASSERT(target);
1054 ASSERT(writes.isEmpty());
1055 if (Node* value = heapResolve(PromotedHeapLocation(target->identifier(), exactRead))) {
1056 ASSERT(!value->replacement());
1057 node->replaceWith(value);
1058 }
1059 Node* identifier = target->get(exactRead);
1060 if (identifier)
1061 m_heap.newPointer(node, identifier);
1062 }
1063
1064 for (auto entry : writes) {
1065 ASSERT(target);
1066 if (entry.value.isNode())
1067 target->set(entry.key, m_heap.follow(entry.value.asNode()));
1068 else
1069 target->remove(entry.key);
1070 heapWrite(PromotedHeapLocation(target->identifier(), entry.key), entry.value);
1071 }
1072
1073 m_heap.assertIsValid();
fpizlo@apple.comfc70ba62014-09-26 03:59:33 +00001074 }
basile_clement@apple.comd7930292015-07-13 23:27:30 +00001075
1076 bool determineSinkCandidates()
fpizlo@apple.comfc70ba62014-09-26 03:59:33 +00001077 {
basile_clement@apple.comd7930292015-07-13 23:27:30 +00001078 m_sinkCandidates.clear();
1079 m_materializationToEscapee.clear();
1080 m_materializationSiteToMaterializations.clear();
1081 m_materializationSiteToRecoveries.clear();
1082
1083 // Logically we wish to consider every allocation and sink
1084 // it. However, it is probably not profitable to sink an
1085 // allocation that will always escape. So, we only sink an
1086 // allocation if one of the following is true:
1087 //
1088 // 1) There exists a basic block with only backwards outgoing
1089 // edges (or no outgoing edges) in which the node wasn't
1090 // materialized. This is meant to catch
1091 // effectively-infinite loops in which we don't need to
1092 // have allocated the object.
1093 //
1094 // 2) There exists a basic block at the tail of which the node
1095 // is dead and not materialized.
1096 //
1097 // 3) The sum of execution counts of the materializations is
1098 // less than the sum of execution counts of the original
1099 // node.
1100 //
1101 // We currently implement only rule #2.
1102 // FIXME: Implement the two other rules.
1103 // https://bugs.webkit.org/show_bug.cgi?id=137073 (rule #1)
1104 // https://bugs.webkit.org/show_bug.cgi?id=137074 (rule #3)
1105 //
1106 // However, these rules allow for a sunk object to be put into
1107 // a non-sunk one, which we don't support. We could solve this
1108 // by supporting PutHints on local allocations, making these
1109 // objects only partially correct, and we would need to adapt
1110 // the OSR availability analysis and OSR exit to handle
1111 // this. This would be totally doable, but would create a
1112 // super rare, and thus bug-prone, code path.
1113 // So, instead, we need to implement one of the following
1114 // closure rules:
1115 //
1116 // 1) If we put a sink candidate into a local allocation that
1117 // is not a sink candidate, change our minds and don't
1118 // actually sink the sink candidate.
1119 //
1120 // 2) If we put a sink candidate into a local allocation, that
1121 // allocation becomes a sink candidate as well.
1122 //
1123 // We currently choose to implement closure rule #2.
1124 HashMap<Node*, Vector<Node*>> dependencies;
1125 bool hasUnescapedReads = false;
1126 for (BasicBlock* block : m_graph.blocksInPreOrder()) {
1127 m_heap = m_heapAtHead[block];
1128
1129 for (Node* node : *block) {
1130 handleNode(
1131 node,
1132 [&] (PromotedHeapLocation location, LazyNode value) {
1133 if (!value.isNode())
1134 return;
1135
1136 Allocation* allocation = m_heap.onlyLocalAllocation(value.asNode());
1137 if (allocation && !allocation->isEscapedAllocation())
1138 dependencies.add(allocation->identifier(), Vector<Node*>()).iterator->value.append(location.base());
1139 },
1140 [&] (PromotedHeapLocation) -> Node* {
1141 hasUnescapedReads = true;
1142 return nullptr;
1143 });
1144 }
1145
1146 // The sink candidates are initially the unescaped
1147 // allocations dying at tail of blocks
1148 HashSet<Node*> allocations;
1149 for (const auto& entry : m_heap.allocations()) {
1150 if (!entry.value.isEscapedAllocation())
1151 allocations.add(entry.key);
1152 }
1153
1154 m_heap.pruneByLiveness(m_combinedLiveness.liveAtTail[block]);
1155
1156 for (Node* identifier : allocations) {
1157 if (!m_heap.isAllocation(identifier))
1158 m_sinkCandidates.add(identifier);
1159 }
1160 }
1161
1162 // Ensure that the set of sink candidates is closed for put operations
1163 Vector<Node*> worklist;
1164 worklist.appendRange(m_sinkCandidates.begin(), m_sinkCandidates.end());
1165
1166 while (!worklist.isEmpty()) {
1167 for (Node* identifier : dependencies.get(worklist.takeLast())) {
1168 if (m_sinkCandidates.add(identifier).isNewEntry)
1169 worklist.append(identifier);
1170 }
1171 }
1172
1173 if (m_sinkCandidates.isEmpty())
1174 return hasUnescapedReads;
1175
1176 if (verbose)
1177 dataLog("Candidates: ", listDump(m_sinkCandidates), "\n");
1178
1179 // Create the materialization nodes
1180 for (BasicBlock* block : m_graph.blocksInNaturalOrder()) {
1181 m_heap = m_heapAtHead[block];
1182 m_heap.setWantEscapees();
1183
1184 for (Node* node : *block) {
1185 handleNode(
1186 node,
1187 [] (PromotedHeapLocation, LazyNode) { },
1188 [] (PromotedHeapLocation) -> Node* {
1189 return nullptr;
1190 });
1191 auto escapees = m_heap.takeEscapees();
1192 if (!escapees.isEmpty())
1193 placeMaterializations(escapees, node);
1194 }
1195
1196 m_heap.pruneByLiveness(m_combinedLiveness.liveAtTail[block]);
1197
1198 {
1199 HashMap<Node*, Allocation> escapingOnEdge;
1200 for (const auto& entry : m_heap.allocations()) {
1201 if (entry.value.isEscapedAllocation())
1202 continue;
1203
1204 bool mustEscape = false;
1205 for (BasicBlock* successorBlock : block->successors()) {
1206 if (!m_heapAtHead[successorBlock].isAllocation(entry.key)
1207 || m_heapAtHead[successorBlock].getAllocation(entry.key).isEscapedAllocation())
1208 mustEscape = true;
1209 }
1210
1211 if (mustEscape)
1212 escapingOnEdge.add(entry.key, entry.value);
1213 }
aestes@apple.com13aae082016-01-02 08:03:08 +00001214 placeMaterializations(WTFMove(escapingOnEdge), block->terminal());
basile_clement@apple.comd7930292015-07-13 23:27:30 +00001215 }
1216 }
1217
1218 return hasUnescapedReads || !m_sinkCandidates.isEmpty();
1219 }
1220
1221 void placeMaterializations(HashMap<Node*, Allocation> escapees, Node* where)
1222 {
1223 // We don't create materializations if the escapee is not a
1224 // sink candidate
1225 Vector<Node*> toRemove;
1226 for (const auto& entry : escapees) {
1227 if (!m_sinkCandidates.contains(entry.key))
1228 toRemove.append(entry.key);
1229 }
1230 for (Node* identifier : toRemove)
1231 escapees.remove(identifier);
1232
1233 if (escapees.isEmpty())
1234 return;
1235
1236 // First collect the hints that will be needed when the node
1237 // we materialize is still stored into other unescaped sink candidates
1238 Vector<PromotedHeapLocation> hints;
1239 for (const auto& entry : m_heap.allocations()) {
1240 if (escapees.contains(entry.key))
1241 continue;
1242
1243 for (const auto& field : entry.value.fields()) {
1244 ASSERT(m_sinkCandidates.contains(entry.key) || !escapees.contains(field.value));
fpizlo@apple.comba93cdc2016-09-01 18:55:50 +00001245 if (escapees.contains(field.value))
basile_clement@apple.comd7930292015-07-13 23:27:30 +00001246 hints.append(PromotedHeapLocation(entry.key, field.key));
1247 }
1248 }
1249
1250 // Now we need to order the materialization. Any order is
1251 // valid (as long as we materialize a node first if it is
1252 // needed for the materialization of another node, e.g. a
1253 // function's activation must be materialized before the
1254 // function itself), but we want to try minimizing the number
1255 // of times we have to place Puts to close cycles after a
1256 // materialization. In other words, we are trying to find the
1257 // minimum number of materializations to remove from the
1258 // materialization graph to make it a DAG, known as the
1259 // (vertex) feedback set problem. Unfortunately, this is a
1260 // NP-hard problem, which we don't want to solve exactly.
1261 //
1262 // Instead, we use a simple greedy procedure, that procedes as
1263 // follow:
1264 // - While there is at least one node with no outgoing edge
1265 // amongst the remaining materializations, materialize it
1266 // first
1267 //
1268 // - Similarily, while there is at least one node with no
1269 // incoming edge amongst the remaining materializations,
1270 // materialize it last.
1271 //
1272 // - When both previous conditions are false, we have an
1273 // actual cycle, and we need to pick a node to
1274 // materialize. We try greedily to remove the "pressure" on
1275 // the remaining nodes by choosing the node with maximum
1276 // |incoming edges| * |outgoing edges| as a measure of how
1277 // "central" to the graph it is. We materialize it first,
1278 // so that all the recoveries will be Puts of things into
1279 // it (rather than Puts of the materialization into other
1280 // objects), which means we will have a single
1281 // StoreBarrier.
1282
1283
1284 // Compute dependencies between materializations
1285 HashMap<Node*, HashSet<Node*>> dependencies;
1286 HashMap<Node*, HashSet<Node*>> reverseDependencies;
1287 HashMap<Node*, HashSet<Node*>> forMaterialization;
1288 for (const auto& entry : escapees) {
1289 auto& myDependencies = dependencies.add(entry.key, HashSet<Node*>()).iterator->value;
1290 auto& myDependenciesForMaterialization = forMaterialization.add(entry.key, HashSet<Node*>()).iterator->value;
1291 reverseDependencies.add(entry.key, HashSet<Node*>());
1292 for (const auto& field : entry.value.fields()) {
1293 if (escapees.contains(field.value) && field.value != entry.key) {
1294 myDependencies.add(field.value);
1295 reverseDependencies.add(field.value, HashSet<Node*>()).iterator->value.add(entry.key);
1296 if (field.key.neededForMaterialization())
1297 myDependenciesForMaterialization.add(field.value);
1298 }
1299 }
1300 }
1301
1302 // Helper function to update the materialized set and the
1303 // dependencies
1304 HashSet<Node*> materialized;
1305 auto materialize = [&] (Node* identifier) {
1306 materialized.add(identifier);
1307 for (Node* dep : dependencies.get(identifier))
1308 reverseDependencies.find(dep)->value.remove(identifier);
1309 for (Node* rdep : reverseDependencies.get(identifier)) {
1310 dependencies.find(rdep)->value.remove(identifier);
1311 forMaterialization.find(rdep)->value.remove(identifier);
1312 }
1313 dependencies.remove(identifier);
1314 reverseDependencies.remove(identifier);
1315 forMaterialization.remove(identifier);
1316 };
1317
1318 // Nodes without remaining unmaterialized fields will be
1319 // materialized first - amongst the remaining unmaterialized
1320 // nodes
1321 std::list<Allocation> toMaterialize;
1322 auto firstPos = toMaterialize.begin();
1323 auto materializeFirst = [&] (Allocation&& allocation) {
1324 materialize(allocation.identifier());
1325 // We need to insert *after* the current position
1326 if (firstPos != toMaterialize.end())
1327 ++firstPos;
aestes@apple.com13aae082016-01-02 08:03:08 +00001328 firstPos = toMaterialize.insert(firstPos, WTFMove(allocation));
basile_clement@apple.comd7930292015-07-13 23:27:30 +00001329 };
1330
1331 // Nodes that no other unmaterialized node points to will be
1332 // materialized last - amongst the remaining unmaterialized
1333 // nodes
1334 auto lastPos = toMaterialize.end();
1335 auto materializeLast = [&] (Allocation&& allocation) {
1336 materialize(allocation.identifier());
aestes@apple.com13aae082016-01-02 08:03:08 +00001337 lastPos = toMaterialize.insert(lastPos, WTFMove(allocation));
basile_clement@apple.comd7930292015-07-13 23:27:30 +00001338 };
1339
1340 // These are the promoted locations that contains some of the
1341 // allocations we are currently escaping. If they are a location on
1342 // some other allocation we are currently materializing, we will need
1343 // to "recover" their value with a real put once the corresponding
1344 // allocation is materialized; if they are a location on some other
1345 // not-yet-materialized allocation, we will need a PutHint.
1346 Vector<PromotedHeapLocation> toRecover;
1347
1348 // This loop does the actual cycle breaking
1349 while (!escapees.isEmpty()) {
1350 materialized.clear();
1351
1352 // Materialize nodes that won't require recoveries if we can
1353 for (auto& entry : escapees) {
1354 if (!forMaterialization.find(entry.key)->value.isEmpty())
1355 continue;
1356
1357 if (dependencies.find(entry.key)->value.isEmpty()) {
aestes@apple.com13aae082016-01-02 08:03:08 +00001358 materializeFirst(WTFMove(entry.value));
basile_clement@apple.comd7930292015-07-13 23:27:30 +00001359 continue;
1360 }
1361
1362 if (reverseDependencies.find(entry.key)->value.isEmpty()) {
aestes@apple.com13aae082016-01-02 08:03:08 +00001363 materializeLast(WTFMove(entry.value));
basile_clement@apple.comd7930292015-07-13 23:27:30 +00001364 continue;
1365 }
1366 }
1367
1368 // We reach this only if there is an actual cycle that needs
1369 // breaking. Because we do not want to solve a NP-hard problem
1370 // here, we just heuristically pick a node and materialize it
1371 // first.
1372 if (materialized.isEmpty()) {
1373 uint64_t maxEvaluation = 0;
peavo@outlook.coma83d48a2016-05-02 21:20:15 +00001374 Allocation* bestAllocation = nullptr;
basile_clement@apple.comd7930292015-07-13 23:27:30 +00001375 for (auto& entry : escapees) {
1376 if (!forMaterialization.find(entry.key)->value.isEmpty())
1377 continue;
1378
1379 uint64_t evaluation =
1380 static_cast<uint64_t>(dependencies.get(entry.key).size()) * reverseDependencies.get(entry.key).size();
1381 if (evaluation > maxEvaluation) {
1382 maxEvaluation = evaluation;
1383 bestAllocation = &entry.value;
1384 }
1385 }
1386 RELEASE_ASSERT(maxEvaluation > 0);
1387
aestes@apple.com13aae082016-01-02 08:03:08 +00001388 materializeFirst(WTFMove(*bestAllocation));
basile_clement@apple.comd7930292015-07-13 23:27:30 +00001389 }
1390 RELEASE_ASSERT(!materialized.isEmpty());
1391
1392 for (Node* identifier : materialized)
1393 escapees.remove(identifier);
1394 }
1395
1396 materialized.clear();
1397
1398 HashSet<Node*> escaped;
1399 for (const Allocation& allocation : toMaterialize)
1400 escaped.add(allocation.identifier());
1401 for (const Allocation& allocation : toMaterialize) {
1402 for (const auto& field : allocation.fields()) {
1403 if (escaped.contains(field.value) && !materialized.contains(field.value))
1404 toRecover.append(PromotedHeapLocation(allocation.identifier(), field.key));
1405 }
1406 materialized.add(allocation.identifier());
1407 }
1408
1409 Vector<Node*>& materializations = m_materializationSiteToMaterializations.add(
1410 where, Vector<Node*>()).iterator->value;
1411
1412 for (const Allocation& allocation : toMaterialize) {
1413 Node* materialization = createMaterialization(allocation, where);
1414 materializations.append(materialization);
1415 m_materializationToEscapee.add(materialization, allocation.identifier());
1416 }
1417
1418 if (!toRecover.isEmpty()) {
1419 m_materializationSiteToRecoveries.add(
1420 where, Vector<PromotedHeapLocation>()).iterator->value.appendVector(toRecover);
1421 }
1422
1423 // The hints need to be after the "real" recoveries so that we
1424 // don't hint not-yet-complete objects
1425 if (!hints.isEmpty()) {
1426 m_materializationSiteToRecoveries.add(
1427 where, Vector<PromotedHeapLocation>()).iterator->value.appendVector(hints);
1428 }
1429 }
1430
1431 Node* createMaterialization(const Allocation& allocation, Node* where)
1432 {
1433 // FIXME: This is the only place where we actually use the
1434 // fact that an allocation's identifier is indeed the node
1435 // that created the allocation.
1436 switch (allocation.kind()) {
1437 case Allocation::Kind::Object: {
fpizlo@apple.comfc70ba62014-09-26 03:59:33 +00001438 ObjectMaterializationData* data = m_graph.m_objectMaterializationData.add();
basile_clement@apple.comd7930292015-07-13 23:27:30 +00001439 StructureSet* set = m_graph.addStructureSet(allocation.structures());
1440
1441 return m_graph.addNode(
1442 allocation.identifier()->prediction(), Node::VarArg, MaterializeNewObject,
fpizlo@apple.com583e2f02015-08-21 20:48:59 +00001443 where->origin.withSemantic(allocation.identifier()->origin.semantic),
basile_clement@apple.comd7930292015-07-13 23:27:30 +00001444 OpInfo(set), OpInfo(data), 0, 0);
1445 }
1446
utatane.tea@gmail.com9d9fd322015-12-17 10:33:08 +00001447 case Allocation::Kind::GeneratorFunction:
basile_clement@apple.comd7930292015-07-13 23:27:30 +00001448 case Allocation::Kind::Function: {
1449 FrozenValue* executable = allocation.identifier()->cellOperand();
commit-queue@webkit.orga4201b02015-08-17 22:24:20 +00001450
utatane.tea@gmail.com9d9fd322015-12-17 10:33:08 +00001451 NodeType nodeType =
utatane.tea@gmail.com9d9fd322015-12-17 10:33:08 +00001452 allocation.kind() == Allocation::Kind::GeneratorFunction ? NewGeneratorFunction : NewFunction;
commit-queue@webkit.orga4201b02015-08-17 22:24:20 +00001453
basile_clement@apple.comd7930292015-07-13 23:27:30 +00001454 return m_graph.addNode(
commit-queue@webkit.orga4201b02015-08-17 22:24:20 +00001455 allocation.identifier()->prediction(), nodeType,
fpizlo@apple.com583e2f02015-08-21 20:48:59 +00001456 where->origin.withSemantic(
1457 allocation.identifier()->origin.semantic),
basile_clement@apple.comd7930292015-07-13 23:27:30 +00001458 OpInfo(executable));
fpizlo@apple.com6fa73b52014-10-02 19:38:08 +00001459 break;
fpizlo@apple.comfc70ba62014-09-26 03:59:33 +00001460 }
commit-queue@webkit.org88ab4e72015-04-24 02:23:36 +00001461
basile_clement@apple.comd7930292015-07-13 23:27:30 +00001462 case Allocation::Kind::Activation: {
basile_clement@apple.com2ca1f7b2015-05-05 16:34:21 +00001463 ObjectMaterializationData* data = m_graph.m_objectMaterializationData.add();
basile_clement@apple.comd7930292015-07-13 23:27:30 +00001464 FrozenValue* symbolTable = allocation.identifier()->cellOperand();
1465
1466 return m_graph.addNode(
1467 allocation.identifier()->prediction(), Node::VarArg, MaterializeCreateActivation,
fpizlo@apple.com583e2f02015-08-21 20:48:59 +00001468 where->origin.withSemantic(
1469 allocation.identifier()->origin.semantic),
basile_clement@apple.comd7930292015-07-13 23:27:30 +00001470 OpInfo(symbolTable), OpInfo(data), 0, 0);
basile_clement@apple.com2ca1f7b2015-05-05 16:34:21 +00001471 }
1472
fpizlo@apple.comfc70ba62014-09-26 03:59:33 +00001473 default:
basile_clement@apple.comd7930292015-07-13 23:27:30 +00001474 DFG_CRASH(m_graph, allocation.identifier(), "Bad allocation kind");
fpizlo@apple.comfc70ba62014-09-26 03:59:33 +00001475 }
basile_clement@apple.comd7930292015-07-13 23:27:30 +00001476 }
1477
1478 void promoteLocalHeap()
1479 {
1480 // Collect the set of heap locations that we will be operating
1481 // over.
1482 HashSet<PromotedHeapLocation> locations;
1483 for (BasicBlock* block : m_graph.blocksInNaturalOrder()) {
1484 m_heap = m_heapAtHead[block];
1485
1486 for (Node* node : *block) {
1487 handleNode(
1488 node,
1489 [&] (PromotedHeapLocation location, LazyNode) {
1490 // If the location is not on a sink candidate,
1491 // we only sink it if it is read
1492 if (m_sinkCandidates.contains(location.base()))
1493 locations.add(location);
1494 },
1495 [&] (PromotedHeapLocation location) -> Node* {
1496 locations.add(location);
1497 return nullptr;
1498 });
1499 }
1500 }
1501
1502 // Figure out which locations belong to which allocations.
1503 m_locationsForAllocation.clear();
1504 for (PromotedHeapLocation location : locations) {
1505 auto result = m_locationsForAllocation.add(
1506 location.base(),
1507 Vector<PromotedHeapLocation>());
1508 ASSERT(!result.iterator->value.contains(location));
1509 result.iterator->value.append(location);
1510 }
1511
1512 m_pointerSSA.reset();
1513 m_allocationSSA.reset();
1514
1515 // Collect the set of "variables" that we will be sinking.
1516 m_locationToVariable.clear();
1517 m_nodeToVariable.clear();
1518 Vector<Node*> indexToNode;
1519 Vector<PromotedHeapLocation> indexToLocation;
1520
1521 for (Node* index : m_sinkCandidates) {
1522 SSACalculator::Variable* variable = m_allocationSSA.newVariable();
1523 m_nodeToVariable.add(index, variable);
1524 ASSERT(indexToNode.size() == variable->index());
1525 indexToNode.append(index);
1526 }
1527
1528 for (PromotedHeapLocation location : locations) {
1529 SSACalculator::Variable* variable = m_pointerSSA.newVariable();
1530 m_locationToVariable.add(location, variable);
1531 ASSERT(indexToLocation.size() == variable->index());
1532 indexToLocation.append(location);
1533 }
1534
1535 // We insert all required constants at top of block 0 so that
1536 // they are inserted only once and we don't clutter the graph
1537 // with useless constants everywhere
1538 HashMap<FrozenValue*, Node*> lazyMapping;
1539 if (!m_bottom)
fpizlo@apple.comc9d072f2015-08-27 07:16:07 +00001540 m_bottom = m_insertionSet.insertConstant(0, m_graph.block(0)->at(0)->origin, jsNumber(1927));
basile_clement@apple.comd7930292015-07-13 23:27:30 +00001541 for (BasicBlock* block : m_graph.blocksInNaturalOrder()) {
1542 m_heap = m_heapAtHead[block];
1543
1544 for (unsigned nodeIndex = 0; nodeIndex < block->size(); ++nodeIndex) {
1545 Node* node = block->at(nodeIndex);
1546
1547 // Some named properties can be added conditionally,
1548 // and that would necessitate bottoms
1549 for (PromotedHeapLocation location : m_locationsForAllocation.get(node)) {
1550 if (location.kind() != NamedPropertyPLoc)
1551 continue;
1552
1553 SSACalculator::Variable* variable = m_locationToVariable.get(location);
1554 m_pointerSSA.newDef(variable, block, m_bottom);
1555 }
1556
1557 for (Node* materialization : m_materializationSiteToMaterializations.get(node)) {
1558 Node* escapee = m_materializationToEscapee.get(materialization);
1559 m_allocationSSA.newDef(m_nodeToVariable.get(escapee), block, materialization);
1560 }
1561
1562 if (m_sinkCandidates.contains(node))
1563 m_allocationSSA.newDef(m_nodeToVariable.get(node), block, node);
1564
1565 handleNode(
1566 node,
1567 [&] (PromotedHeapLocation location, LazyNode value) {
1568 if (!locations.contains(location))
1569 return;
1570
1571 Node* nodeValue;
1572 if (value.isNode())
1573 nodeValue = value.asNode();
1574 else {
1575 auto iter = lazyMapping.find(value.asValue());
1576 if (iter != lazyMapping.end())
1577 nodeValue = iter->value;
1578 else {
1579 nodeValue = value.ensureIsNode(
1580 m_insertionSet, m_graph.block(0), 0);
1581 lazyMapping.add(value.asValue(), nodeValue);
1582 }
1583 }
1584
1585 SSACalculator::Variable* variable = m_locationToVariable.get(location);
1586 m_pointerSSA.newDef(variable, block, nodeValue);
1587 },
1588 [] (PromotedHeapLocation) -> Node* {
1589 return nullptr;
1590 });
1591 }
1592 }
1593 m_insertionSet.execute(m_graph.block(0));
1594
1595 // Run the SSA calculators to create Phis
1596 m_pointerSSA.computePhis(
1597 [&] (SSACalculator::Variable* variable, BasicBlock* block) -> Node* {
1598 PromotedHeapLocation location = indexToLocation[variable->index()];
1599
1600 // Don't create Phi nodes for fields of dead allocations
1601 if (!m_heapAtHead[block].isAllocation(location.base()))
1602 return nullptr;
1603
1604 // Don't create Phi nodes once we are escaped
1605 if (m_heapAtHead[block].getAllocation(location.base()).isEscapedAllocation())
1606 return nullptr;
1607
1608 // If we point to a single allocation, we will
1609 // directly use its materialization
1610 if (m_heapAtHead[block].follow(location))
1611 return nullptr;
1612
fpizlo@apple.comc9d072f2015-08-27 07:16:07 +00001613 Node* phiNode = m_graph.addNode(SpecHeapTop, Phi, block->at(0)->origin.withInvalidExit());
basile_clement@apple.comd7930292015-07-13 23:27:30 +00001614 phiNode->mergeFlags(NodeResultJS);
1615 return phiNode;
1616 });
1617
1618 m_allocationSSA.computePhis(
1619 [&] (SSACalculator::Variable* variable, BasicBlock* block) -> Node* {
1620 Node* identifier = indexToNode[variable->index()];
1621
1622 // Don't create Phi nodes for dead allocations
1623 if (!m_heapAtHead[block].isAllocation(identifier))
1624 return nullptr;
1625
1626 // Don't create Phi nodes until we are escaped
1627 if (!m_heapAtHead[block].getAllocation(identifier).isEscapedAllocation())
1628 return nullptr;
1629
fpizlo@apple.comc9d072f2015-08-27 07:16:07 +00001630 Node* phiNode = m_graph.addNode(SpecHeapTop, Phi, block->at(0)->origin.withInvalidExit());
basile_clement@apple.comd7930292015-07-13 23:27:30 +00001631 phiNode->mergeFlags(NodeResultJS);
1632 return phiNode;
1633 });
1634
1635 // Place Phis in the right places, replace all uses of any load with the appropriate
1636 // value, and create the materialization nodes.
1637 LocalOSRAvailabilityCalculator availabilityCalculator;
1638 m_graph.clearReplacements();
1639 for (BasicBlock* block : m_graph.blocksInPreOrder()) {
1640 m_heap = m_heapAtHead[block];
1641 availabilityCalculator.beginBlock(block);
1642
1643 // These mapping tables are intended to be lazy. If
1644 // something is omitted from the table, it means that
1645 // there haven't been any local stores to the promoted
1646 // heap location (or any local materialization).
1647 m_localMapping.clear();
1648 m_escapeeToMaterialization.clear();
1649
1650 // Insert the Phi functions that we had previously
1651 // created.
1652 for (SSACalculator::Def* phiDef : m_pointerSSA.phisForBlock(block)) {
1653 SSACalculator::Variable* variable = phiDef->variable();
1654 m_insertionSet.insert(0, phiDef->value());
1655
1656 PromotedHeapLocation location = indexToLocation[variable->index()];
1657 m_localMapping.set(location, phiDef->value());
1658
1659 if (m_sinkCandidates.contains(location.base())) {
1660 m_insertionSet.insert(
fpizlo@apple.comc9d072f2015-08-27 07:16:07 +00001661 0,
1662 location.createHint(
1663 m_graph, block->at(0)->origin.withInvalidExit(), phiDef->value()));
basile_clement@apple.comd7930292015-07-13 23:27:30 +00001664 }
1665 }
1666
1667 for (SSACalculator::Def* phiDef : m_allocationSSA.phisForBlock(block)) {
1668 SSACalculator::Variable* variable = phiDef->variable();
1669 m_insertionSet.insert(0, phiDef->value());
1670
1671 Node* identifier = indexToNode[variable->index()];
1672 m_escapeeToMaterialization.add(identifier, phiDef->value());
fpizlo@apple.comf29186e2015-08-26 19:24:41 +00001673 bool canExit = false;
fpizlo@apple.comc9d072f2015-08-27 07:16:07 +00001674 insertOSRHintsForUpdate(
1675 0, block->at(0)->origin, canExit,
1676 availabilityCalculator.m_availability, identifier, phiDef->value());
basile_clement@apple.comd7930292015-07-13 23:27:30 +00001677 }
1678
1679 if (verbose) {
1680 dataLog("Local mapping at ", pointerDump(block), ": ", mapDump(m_localMapping), "\n");
1681 dataLog("Local materializations at ", pointerDump(block), ": ", mapDump(m_escapeeToMaterialization), "\n");
1682 }
1683
1684 for (unsigned nodeIndex = 0; nodeIndex < block->size(); ++nodeIndex) {
1685 Node* node = block->at(nodeIndex);
fpizlo@apple.comf29186e2015-08-26 19:24:41 +00001686 bool canExit = true;
1687 bool nextCanExit = node->origin.exitOK;
basile_clement@apple.comd7930292015-07-13 23:27:30 +00001688 for (PromotedHeapLocation location : m_locationsForAllocation.get(node)) {
1689 if (location.kind() != NamedPropertyPLoc)
1690 continue;
1691
1692 m_localMapping.set(location, m_bottom);
1693
1694 if (m_sinkCandidates.contains(node)) {
fpizlo@apple.comba93cdc2016-09-01 18:55:50 +00001695 if (verbose)
1696 dataLog("For sink candidate ", node, " found location ", location, "\n");
basile_clement@apple.comd7930292015-07-13 23:27:30 +00001697 m_insertionSet.insert(
1698 nodeIndex + 1,
fpizlo@apple.comf29186e2015-08-26 19:24:41 +00001699 location.createHint(
1700 m_graph, node->origin.takeValidExit(nextCanExit), m_bottom));
basile_clement@apple.comd7930292015-07-13 23:27:30 +00001701 }
1702 }
1703
1704 for (Node* materialization : m_materializationSiteToMaterializations.get(node)) {
fpizlo@apple.comf29186e2015-08-26 19:24:41 +00001705 materialization->origin.exitOK &= canExit;
basile_clement@apple.comd7930292015-07-13 23:27:30 +00001706 Node* escapee = m_materializationToEscapee.get(materialization);
1707 populateMaterialization(block, materialization, escapee);
1708 m_escapeeToMaterialization.set(escapee, materialization);
1709 m_insertionSet.insert(nodeIndex, materialization);
1710 if (verbose)
1711 dataLog("Materializing ", escapee, " => ", materialization, " at ", node, "\n");
1712 }
1713
1714 for (PromotedHeapLocation location : m_materializationSiteToRecoveries.get(node))
fpizlo@apple.comf29186e2015-08-26 19:24:41 +00001715 m_insertionSet.insert(nodeIndex, createRecovery(block, location, node, canExit));
basile_clement@apple.comd7930292015-07-13 23:27:30 +00001716
1717 // We need to put the OSR hints after the recoveries,
1718 // because we only want the hints once the object is
1719 // complete
1720 for (Node* materialization : m_materializationSiteToMaterializations.get(node)) {
1721 Node* escapee = m_materializationToEscapee.get(materialization);
1722 insertOSRHintsForUpdate(
fpizlo@apple.comf29186e2015-08-26 19:24:41 +00001723 nodeIndex, node->origin, canExit,
basile_clement@apple.comd7930292015-07-13 23:27:30 +00001724 availabilityCalculator.m_availability, escapee, materialization);
1725 }
1726
fpizlo@apple.comf29186e2015-08-26 19:24:41 +00001727 if (node->origin.exitOK && !canExit) {
1728 // We indicate that the exit state is fine now. It is OK because we updated the
1729 // state above. We need to indicate this manually because the validation doesn't
1730 // have enough information to infer that the exit state is fine.
1731 m_insertionSet.insertNode(nodeIndex, SpecNone, ExitOK, node->origin);
1732 }
1733
basile_clement@apple.comd7930292015-07-13 23:27:30 +00001734 if (m_sinkCandidates.contains(node))
1735 m_escapeeToMaterialization.set(node, node);
1736
1737 availabilityCalculator.executeNode(node);
1738
fpizlo@apple.comf29186e2015-08-26 19:24:41 +00001739 bool desiredNextExitOK = node->origin.exitOK && !clobbersExitState(m_graph, node);
1740
basile_clement@apple.comd7930292015-07-13 23:27:30 +00001741 bool doLower = false;
1742 handleNode(
1743 node,
1744 [&] (PromotedHeapLocation location, LazyNode value) {
1745 if (!locations.contains(location))
1746 return;
1747
1748 Node* nodeValue;
1749 if (value.isNode())
1750 nodeValue = value.asNode();
1751 else
1752 nodeValue = lazyMapping.get(value.asValue());
1753
1754 nodeValue = resolve(block, nodeValue);
1755
1756 m_localMapping.set(location, nodeValue);
1757
1758 if (!m_sinkCandidates.contains(location.base()))
1759 return;
1760
1761 doLower = true;
1762
fpizlo@apple.comba93cdc2016-09-01 18:55:50 +00001763 if (verbose)
1764 dataLog("Creating hint with value ", nodeValue, " before ", node, "\n");
fpizlo@apple.comf29186e2015-08-26 19:24:41 +00001765 m_insertionSet.insert(
1766 nodeIndex + 1,
1767 location.createHint(
1768 m_graph, node->origin.takeValidExit(nextCanExit), nodeValue));
basile_clement@apple.comd7930292015-07-13 23:27:30 +00001769 },
1770 [&] (PromotedHeapLocation location) -> Node* {
1771 return resolve(block, location);
1772 });
1773
fpizlo@apple.comf29186e2015-08-26 19:24:41 +00001774 if (!nextCanExit && desiredNextExitOK) {
1775 // We indicate that the exit state is fine now. We need to do this because we
1776 // emitted hints that appear to invalidate the exit state.
1777 m_insertionSet.insertNode(nodeIndex + 1, SpecNone, ExitOK, node->origin);
1778 }
1779
basile_clement@apple.comd7930292015-07-13 23:27:30 +00001780 if (m_sinkCandidates.contains(node) || doLower) {
1781 switch (node->op()) {
1782 case NewObject:
basile_clement@apple.comd7930292015-07-13 23:27:30 +00001783 node->convertToPhantomNewObject();
1784 break;
1785
1786 case NewFunction:
1787 node->convertToPhantomNewFunction();
1788 break;
1789
utatane.tea@gmail.com9d9fd322015-12-17 10:33:08 +00001790 case NewGeneratorFunction:
1791 node->convertToPhantomNewGeneratorFunction();
1792 break;
1793
basile_clement@apple.comd7930292015-07-13 23:27:30 +00001794 case CreateActivation:
basile_clement@apple.comd7930292015-07-13 23:27:30 +00001795 node->convertToPhantomCreateActivation();
1796 break;
1797
1798 default:
1799 node->remove();
1800 break;
1801 }
1802 }
1803
1804 m_graph.doToChildren(
1805 node,
1806 [&] (Edge& edge) {
1807 edge.setNode(resolve(block, edge.node()));
1808 });
1809 }
1810
1811 // Gotta drop some Upsilons.
1812 NodeAndIndex terminal = block->findTerminal();
1813 size_t upsilonInsertionPoint = terminal.index;
1814 NodeOrigin upsilonOrigin = terminal.node->origin;
1815 for (BasicBlock* successorBlock : block->successors()) {
1816 for (SSACalculator::Def* phiDef : m_pointerSSA.phisForBlock(successorBlock)) {
1817 Node* phiNode = phiDef->value();
1818 SSACalculator::Variable* variable = phiDef->variable();
1819 PromotedHeapLocation location = indexToLocation[variable->index()];
1820 Node* incoming = resolve(block, location);
1821
1822 m_insertionSet.insertNode(
1823 upsilonInsertionPoint, SpecNone, Upsilon, upsilonOrigin,
1824 OpInfo(phiNode), incoming->defaultEdge());
1825 }
1826
1827 for (SSACalculator::Def* phiDef : m_allocationSSA.phisForBlock(successorBlock)) {
1828 Node* phiNode = phiDef->value();
1829 SSACalculator::Variable* variable = phiDef->variable();
1830 Node* incoming = getMaterialization(block, indexToNode[variable->index()]);
1831
1832 m_insertionSet.insertNode(
1833 upsilonInsertionPoint, SpecNone, Upsilon, upsilonOrigin,
1834 OpInfo(phiNode), incoming->defaultEdge());
1835 }
1836 }
1837
1838 m_insertionSet.execute(block);
1839 }
1840 }
1841
1842 Node* resolve(BasicBlock* block, PromotedHeapLocation location)
1843 {
1844 // If we are currently pointing to a single local allocation,
1845 // simply return the associated materialization.
1846 if (Node* identifier = m_heap.follow(location))
1847 return getMaterialization(block, identifier);
1848
1849 if (Node* result = m_localMapping.get(location))
1850 return result;
1851
1852 // This implies that there is no local mapping. Find a non-local mapping.
1853 SSACalculator::Def* def = m_pointerSSA.nonLocalReachingDef(
1854 block, m_locationToVariable.get(location));
1855 ASSERT(def);
1856 ASSERT(def->value());
1857
1858 Node* result = def->value();
sbarati@apple.comc24c3232016-04-10 00:26:25 +00001859 if (result->replacement())
1860 result = result->replacement();
basile_clement@apple.comd7930292015-07-13 23:27:30 +00001861 ASSERT(!result->replacement());
1862
1863 m_localMapping.add(location, result);
fpizlo@apple.com6fa73b52014-10-02 19:38:08 +00001864 return result;
fpizlo@apple.comfc70ba62014-09-26 03:59:33 +00001865 }
basile_clement@apple.comd7930292015-07-13 23:27:30 +00001866
1867 Node* resolve(BasicBlock* block, Node* node)
fpizlo@apple.comfc70ba62014-09-26 03:59:33 +00001868 {
basile_clement@apple.comd7930292015-07-13 23:27:30 +00001869 // If we are currently pointing to a single local allocation,
1870 // simply return the associated materialization.
1871 if (Node* identifier = m_heap.follow(node))
1872 return getMaterialization(block, identifier);
1873
1874 if (node->replacement())
1875 node = node->replacement();
1876 ASSERT(!node->replacement());
1877
1878 return node;
1879 }
1880
1881 Node* getMaterialization(BasicBlock* block, Node* identifier)
1882 {
1883 ASSERT(m_heap.isAllocation(identifier));
1884 if (!m_sinkCandidates.contains(identifier))
1885 return identifier;
1886
1887 if (Node* materialization = m_escapeeToMaterialization.get(identifier))
1888 return materialization;
1889
1890 SSACalculator::Def* def = m_allocationSSA.nonLocalReachingDef(
1891 block, m_nodeToVariable.get(identifier));
1892 ASSERT(def && def->value());
1893 m_escapeeToMaterialization.add(identifier, def->value());
1894 ASSERT(!def->value()->replacement());
1895 return def->value();
1896 }
1897
fpizlo@apple.comf29186e2015-08-26 19:24:41 +00001898 void insertOSRHintsForUpdate(unsigned nodeIndex, NodeOrigin origin, bool& canExit, AvailabilityMap& availability, Node* escapee, Node* materialization)
basile_clement@apple.comd7930292015-07-13 23:27:30 +00001899 {
fpizlo@apple.comba93cdc2016-09-01 18:55:50 +00001900 if (verbose) {
1901 dataLog("Inserting OSR hints at ", origin, ":\n");
1902 dataLog(" Escapee: ", escapee, "\n");
1903 dataLog(" Materialization: ", materialization, "\n");
1904 dataLog(" Availability: ", availability, "\n");
1905 }
1906
basile_clement@apple.comd7930292015-07-13 23:27:30 +00001907 // We need to follow() the value in the heap.
1908 // Consider the following graph:
1909 //
1910 // Block #0
1911 // 0: NewObject({})
1912 // 1: NewObject({})
1913 // -: PutByOffset(@0, @1, x:0)
1914 // -: PutStructure(@0, {x:0})
1915 // 2: GetByOffset(@0, x:0)
1916 // -: MovHint(@2, loc1)
1917 // -: Branch(#1, #2)
1918 //
1919 // Block #1
1920 // 3: Call(f, @1)
1921 // 4: Return(@0)
1922 //
1923 // Block #2
1924 // -: Return(undefined)
1925 //
1926 // We need to materialize @1 at @3, and when doing so we need
1927 // to insert a MovHint for the materialization into loc1 as
1928 // well.
1929 // In order to do this, we say that we need to insert an
1930 // update hint for any availability whose node resolve()s to
1931 // the materialization.
1932 for (auto entry : availability.m_heap) {
1933 if (!entry.value.hasNode())
1934 continue;
1935 if (m_heap.follow(entry.value.node()) != escapee)
1936 continue;
1937
1938 m_insertionSet.insert(
fpizlo@apple.comf29186e2015-08-26 19:24:41 +00001939 nodeIndex,
1940 entry.key.createHint(m_graph, origin.takeValidExit(canExit), materialization));
basile_clement@apple.comd7930292015-07-13 23:27:30 +00001941 }
1942
1943 for (unsigned i = availability.m_locals.size(); i--;) {
1944 if (!availability.m_locals[i].hasNode())
1945 continue;
1946 if (m_heap.follow(availability.m_locals[i].node()) != escapee)
1947 continue;
1948
1949 int operand = availability.m_locals.operandForIndex(i);
1950 m_insertionSet.insertNode(
fpizlo@apple.comf29186e2015-08-26 19:24:41 +00001951 nodeIndex, SpecNone, MovHint, origin.takeValidExit(canExit), OpInfo(operand),
basile_clement@apple.comd7930292015-07-13 23:27:30 +00001952 materialization->defaultEdge());
1953 }
1954 }
1955
1956 void populateMaterialization(BasicBlock* block, Node* node, Node* escapee)
1957 {
1958 Allocation& allocation = m_heap.getAllocation(escapee);
fpizlo@apple.comfc70ba62014-09-26 03:59:33 +00001959 switch (node->op()) {
1960 case MaterializeNewObject: {
1961 ObjectMaterializationData& data = node->objectMaterializationData();
1962 unsigned firstChild = m_graph.m_varArgChildren.size();
basile_clement@apple.comd7930292015-07-13 23:27:30 +00001963
fpizlo@apple.comfc70ba62014-09-26 03:59:33 +00001964 Vector<PromotedHeapLocation> locations = m_locationsForAllocation.get(escapee);
basile_clement@apple.comd7930292015-07-13 23:27:30 +00001965
1966 PromotedHeapLocation structure(StructurePLoc, allocation.identifier());
fpizlo@apple.comfc70ba62014-09-26 03:59:33 +00001967 ASSERT(locations.contains(structure));
basile_clement@apple.comd7930292015-07-13 23:27:30 +00001968
fpizlo@apple.comfc70ba62014-09-26 03:59:33 +00001969 m_graph.m_varArgChildren.append(Edge(resolve(block, structure), KnownCellUse));
basile_clement@apple.comd7930292015-07-13 23:27:30 +00001970
1971 for (PromotedHeapLocation location : locations) {
1972 switch (location.kind()) {
1973 case StructurePLoc:
1974 ASSERT(location == structure);
fpizlo@apple.comfc70ba62014-09-26 03:59:33 +00001975 break;
basile_clement@apple.comd7930292015-07-13 23:27:30 +00001976
fpizlo@apple.comfc70ba62014-09-26 03:59:33 +00001977 case NamedPropertyPLoc: {
basile_clement@apple.comd7930292015-07-13 23:27:30 +00001978 ASSERT(location.base() == allocation.identifier());
fpizlo@apple.com280ef002016-04-05 22:13:16 +00001979 data.m_properties.append(location.descriptor());
basile_clement@apple.comd7930292015-07-13 23:27:30 +00001980 Node* value = resolve(block, location);
1981 if (m_sinkCandidates.contains(value))
1982 m_graph.m_varArgChildren.append(m_bottom);
1983 else
1984 m_graph.m_varArgChildren.append(value);
fpizlo@apple.comfc70ba62014-09-26 03:59:33 +00001985 break;
1986 }
basile_clement@apple.comd7930292015-07-13 23:27:30 +00001987
fpizlo@apple.comfc70ba62014-09-26 03:59:33 +00001988 default:
1989 DFG_CRASH(m_graph, node, "Bad location kind");
1990 }
1991 }
basile_clement@apple.comd7930292015-07-13 23:27:30 +00001992
fpizlo@apple.comfc70ba62014-09-26 03:59:33 +00001993 node->children = AdjacencyList(
1994 AdjacencyList::Variable,
1995 firstChild, m_graph.m_varArgChildren.size() - firstChild);
1996 break;
1997 }
commit-queue@webkit.org88ab4e72015-04-24 02:23:36 +00001998
basile_clement@apple.com2ca1f7b2015-05-05 16:34:21 +00001999 case MaterializeCreateActivation: {
2000 ObjectMaterializationData& data = node->objectMaterializationData();
2001
2002 unsigned firstChild = m_graph.m_varArgChildren.size();
2003
2004 Vector<PromotedHeapLocation> locations = m_locationsForAllocation.get(escapee);
2005
basile_clement@apple.comd7930292015-07-13 23:27:30 +00002006 PromotedHeapLocation symbolTable(ActivationSymbolTablePLoc, allocation.identifier());
2007 ASSERT(locations.contains(symbolTable));
2008 ASSERT(node->cellOperand() == resolve(block, symbolTable)->constant());
2009 m_graph.m_varArgChildren.append(Edge(resolve(block, symbolTable), KnownCellUse));
basile_clement@apple.com2ca1f7b2015-05-05 16:34:21 +00002010
basile_clement@apple.comd7930292015-07-13 23:27:30 +00002011 PromotedHeapLocation scope(ActivationScopePLoc, allocation.identifier());
2012 ASSERT(locations.contains(scope));
basile_clement@apple.com2ca1f7b2015-05-05 16:34:21 +00002013 m_graph.m_varArgChildren.append(Edge(resolve(block, scope), KnownCellUse));
2014
basile_clement@apple.comd7930292015-07-13 23:27:30 +00002015 for (PromotedHeapLocation location : locations) {
2016 switch (location.kind()) {
basile_clement@apple.com2ca1f7b2015-05-05 16:34:21 +00002017 case ActivationScopePLoc: {
basile_clement@apple.comd7930292015-07-13 23:27:30 +00002018 ASSERT(location == scope);
basile_clement@apple.com2ca1f7b2015-05-05 16:34:21 +00002019 break;
2020 }
2021
saambarati1@gmail.com402b5852015-05-22 02:39:25 +00002022 case ActivationSymbolTablePLoc: {
basile_clement@apple.comd7930292015-07-13 23:27:30 +00002023 ASSERT(location == symbolTable);
saambarati1@gmail.com402b5852015-05-22 02:39:25 +00002024 break;
2025 }
2026
basile_clement@apple.com2ca1f7b2015-05-05 16:34:21 +00002027 case ClosureVarPLoc: {
basile_clement@apple.comd7930292015-07-13 23:27:30 +00002028 ASSERT(location.base() == allocation.identifier());
fpizlo@apple.com280ef002016-04-05 22:13:16 +00002029 data.m_properties.append(location.descriptor());
basile_clement@apple.comd7930292015-07-13 23:27:30 +00002030 Node* value = resolve(block, location);
2031 if (m_sinkCandidates.contains(value))
2032 m_graph.m_varArgChildren.append(m_bottom);
2033 else
2034 m_graph.m_varArgChildren.append(value);
basile_clement@apple.com2ca1f7b2015-05-05 16:34:21 +00002035 break;
2036 }
2037
2038 default:
2039 DFG_CRASH(m_graph, node, "Bad location kind");
2040 }
2041 }
2042
2043 node->children = AdjacencyList(
2044 AdjacencyList::Variable,
2045 firstChild, m_graph.m_varArgChildren.size() - firstChild);
2046 break;
2047 }
commit-queue@webkit.orga4201b02015-08-17 22:24:20 +00002048
2049 case NewFunction:
utatane.tea@gmail.com9d9fd322015-12-17 10:33:08 +00002050 case NewGeneratorFunction: {
basile_clement@apple.comd7930292015-07-13 23:27:30 +00002051 Vector<PromotedHeapLocation> locations = m_locationsForAllocation.get(escapee);
commit-queue@webkit.orgb2610c02015-12-08 20:24:04 +00002052 ASSERT(locations.size() == 2);
commit-queue@webkit.orga4201b02015-08-17 22:24:20 +00002053
basile_clement@apple.comd7930292015-07-13 23:27:30 +00002054 PromotedHeapLocation executable(FunctionExecutablePLoc, allocation.identifier());
2055 ASSERT_UNUSED(executable, locations.contains(executable));
commit-queue@webkit.orga4201b02015-08-17 22:24:20 +00002056
basile_clement@apple.comd7930292015-07-13 23:27:30 +00002057 PromotedHeapLocation activation(FunctionActivationPLoc, allocation.identifier());
2058 ASSERT(locations.contains(activation));
commit-queue@webkit.org88ab4e72015-04-24 02:23:36 +00002059
basile_clement@apple.comd7930292015-07-13 23:27:30 +00002060 node->child1() = Edge(resolve(block, activation), KnownCellUse);
commit-queue@webkit.org88ab4e72015-04-24 02:23:36 +00002061 break;
2062 }
2063
fpizlo@apple.comfc70ba62014-09-26 03:59:33 +00002064 default:
2065 DFG_CRASH(m_graph, node, "Bad materialize op");
basile_clement@apple.comd7930292015-07-13 23:27:30 +00002066 }
2067 }
2068
fpizlo@apple.comf29186e2015-08-26 19:24:41 +00002069 Node* createRecovery(BasicBlock* block, PromotedHeapLocation location, Node* where, bool& canExit)
basile_clement@apple.comd7930292015-07-13 23:27:30 +00002070 {
2071 if (verbose)
2072 dataLog("Recovering ", location, " at ", where, "\n");
2073 ASSERT(location.base()->isPhantomAllocation());
2074 Node* base = getMaterialization(block, location.base());
2075 Node* value = resolve(block, location);
2076
fpizlo@apple.comf29186e2015-08-26 19:24:41 +00002077 NodeOrigin origin = where->origin.withSemantic(base->origin.semantic);
2078
basile_clement@apple.comd7930292015-07-13 23:27:30 +00002079 if (verbose)
2080 dataLog("Base is ", base, " and value is ", value, "\n");
2081
2082 if (base->isPhantomAllocation()) {
2083 return PromotedHeapLocation(base, location.descriptor()).createHint(
fpizlo@apple.comf29186e2015-08-26 19:24:41 +00002084 m_graph, origin.takeValidExit(canExit), value);
basile_clement@apple.comd7930292015-07-13 23:27:30 +00002085 }
2086
2087 switch (location.kind()) {
2088 case NamedPropertyPLoc: {
2089 Allocation& allocation = m_heap.getAllocation(location.base());
2090
2091 Vector<Structure*> structures;
2092 structures.appendRange(allocation.structures().begin(), allocation.structures().end());
2093 unsigned identifierNumber = location.info();
2094 UniquedStringImpl* uid = m_graph.identifiers()[identifierNumber];
2095
2096 std::sort(
2097 structures.begin(),
2098 structures.end(),
2099 [uid] (Structure *a, Structure* b) -> bool {
2100 return a->getConcurrently(uid) < b->getConcurrently(uid);
2101 });
2102
2103 PropertyOffset firstOffset = structures[0]->getConcurrently(uid);
2104
2105 if (firstOffset == structures.last()->getConcurrently(uid)) {
2106 Node* storage = base;
2107 // FIXME: When we decide to sink objects with a
2108 // property storage, we should handle non-inline offsets.
2109 RELEASE_ASSERT(isInlineOffset(firstOffset));
2110
2111 StorageAccessData* data = m_graph.m_storageAccessData.add();
2112 data->offset = firstOffset;
2113 data->identifierNumber = identifierNumber;
2114
2115 return m_graph.addNode(
basile_clement@apple.comd7930292015-07-13 23:27:30 +00002116 PutByOffset,
fpizlo@apple.comf29186e2015-08-26 19:24:41 +00002117 origin.takeValidExit(canExit),
basile_clement@apple.comd7930292015-07-13 23:27:30 +00002118 OpInfo(data),
2119 Edge(storage, KnownCellUse),
2120 Edge(base, KnownCellUse),
2121 value->defaultEdge());
2122 }
2123
2124 MultiPutByOffsetData* data = m_graph.m_multiPutByOffsetData.add();
2125 data->identifierNumber = identifierNumber;
2126
2127 {
2128 PropertyOffset currentOffset = firstOffset;
2129 StructureSet currentSet;
2130 for (Structure* structure : structures) {
2131 PropertyOffset offset = structure->getConcurrently(uid);
2132 if (offset != currentOffset) {
fpizlo@apple.com12835772015-09-21 20:49:04 +00002133 // Because our analysis treats MultiPutByOffset like an escape, we only have to
2134 // deal with storing results that would have been previously stored by PutByOffset
2135 // nodes. Those nodes were guarded by the appropriate type checks. This means that
2136 // at this point, we can simply trust that the incoming value has the right type
2137 // for whatever structure we are using.
basile_clement@apple.comd7930292015-07-13 23:27:30 +00002138 data->variants.append(
fpizlo@apple.com12835772015-09-21 20:49:04 +00002139 PutByIdVariant::replace(currentSet, currentOffset, InferredType::Top));
basile_clement@apple.comd7930292015-07-13 23:27:30 +00002140 currentOffset = offset;
2141 currentSet.clear();
2142 }
2143 currentSet.add(structure);
2144 }
fpizlo@apple.com12835772015-09-21 20:49:04 +00002145 data->variants.append(
2146 PutByIdVariant::replace(currentSet, currentOffset, InferredType::Top));
basile_clement@apple.comd7930292015-07-13 23:27:30 +00002147 }
2148
2149 return m_graph.addNode(
basile_clement@apple.comd7930292015-07-13 23:27:30 +00002150 MultiPutByOffset,
fpizlo@apple.comf29186e2015-08-26 19:24:41 +00002151 origin.takeValidExit(canExit),
basile_clement@apple.comd7930292015-07-13 23:27:30 +00002152 OpInfo(data),
2153 Edge(base, KnownCellUse),
2154 value->defaultEdge());
2155 break;
2156 }
2157
2158 case ClosureVarPLoc: {
2159 return m_graph.addNode(
basile_clement@apple.comd7930292015-07-13 23:27:30 +00002160 PutClosureVar,
fpizlo@apple.comf29186e2015-08-26 19:24:41 +00002161 origin.takeValidExit(canExit),
basile_clement@apple.comd7930292015-07-13 23:27:30 +00002162 OpInfo(location.info()),
2163 Edge(base, KnownCellUse),
2164 value->defaultEdge());
2165 break;
2166 }
2167
2168 default:
2169 DFG_CRASH(m_graph, base, "Bad location kind");
fpizlo@apple.comfc70ba62014-09-26 03:59:33 +00002170 break;
2171 }
2172 }
basile_clement@apple.comd7930292015-07-13 23:27:30 +00002173
fpizlo@apple.combe625d52016-03-16 20:12:27 +00002174 // This is a great way of asking value->isStillValid() without having to worry about getting
2175 // different answers. It turns out that this analysis works OK regardless of what this
2176 // returns but breaks badly if this changes its mind for any particular InferredValue. This
2177 // method protects us from that.
2178 bool isStillValid(InferredValue* value)
2179 {
2180 return m_validInferredValues.add(value, value->isStillValid()).iterator->value;
2181 }
2182
basile_clement@apple.comd7930292015-07-13 23:27:30 +00002183 SSACalculator m_pointerSSA;
2184 SSACalculator m_allocationSSA;
fpizlo@apple.comfc70ba62014-09-26 03:59:33 +00002185 HashSet<Node*> m_sinkCandidates;
basile_clement@apple.comd7930292015-07-13 23:27:30 +00002186 HashMap<PromotedHeapLocation, SSACalculator::Variable*> m_locationToVariable;
2187 HashMap<Node*, SSACalculator::Variable*> m_nodeToVariable;
2188 HashMap<PromotedHeapLocation, Node*> m_localMapping;
2189 HashMap<Node*, Node*> m_escapeeToMaterialization;
2190 InsertionSet m_insertionSet;
2191 CombinedLiveness m_combinedLiveness;
2192
fpizlo@apple.combe625d52016-03-16 20:12:27 +00002193 HashMap<InferredValue*, bool> m_validInferredValues;
2194
fpizlo@apple.com6fa73b52014-10-02 19:38:08 +00002195 HashMap<Node*, Node*> m_materializationToEscapee;
2196 HashMap<Node*, Vector<Node*>> m_materializationSiteToMaterializations;
basile_clement@apple.comd7930292015-07-13 23:27:30 +00002197 HashMap<Node*, Vector<PromotedHeapLocation>> m_materializationSiteToRecoveries;
2198
fpizlo@apple.comfc70ba62014-09-26 03:59:33 +00002199 HashMap<Node*, Vector<PromotedHeapLocation>> m_locationsForAllocation;
basile_clement@apple.comd7930292015-07-13 23:27:30 +00002200
2201 BlockMap<LocalHeap> m_heapAtHead;
2202 BlockMap<LocalHeap> m_heapAtTail;
2203 LocalHeap m_heap;
2204
2205 Node* m_bottom = nullptr;
fpizlo@apple.comfc70ba62014-09-26 03:59:33 +00002206};
basile_clement@apple.comd7930292015-07-13 23:27:30 +00002207
2208}
2209
fpizlo@apple.comfc70ba62014-09-26 03:59:33 +00002210bool performObjectAllocationSinking(Graph& graph)
2211{
fpizlo@apple.comfc70ba62014-09-26 03:59:33 +00002212 return runPhase<ObjectAllocationSinkingPhase>(graph);
2213}
2214
2215} } // namespace JSC::DFG
2216
2217#endif // ENABLE(DFG_JIT)