OLD | NEW |
1 // Copyright 2016 the V8 project authors. All rights reserved. | 1 // Copyright 2016 the V8 project authors. All rights reserved. |
2 // Use of this source code is governed by a BSD-style license that can be | 2 // Use of this source code is governed by a BSD-style license that can be |
3 // found in the LICENSE file. | 3 // found in the LICENSE file. |
4 | 4 |
5 #include "src/snapshot/startup-serializer.h" | 5 #include "src/snapshot/startup-serializer.h" |
6 | 6 |
7 #include "src/objects-inl.h" | 7 #include "src/objects-inl.h" |
8 #include "src/v8threads.h" | 8 #include "src/v8threads.h" |
9 | 9 |
10 namespace v8 { | 10 namespace v8 { |
11 namespace internal { | 11 namespace internal { |
12 | 12 |
13 StartupSerializer::StartupSerializer(Isolate* isolate, SnapshotByteSink* sink) | 13 StartupSerializer::StartupSerializer( |
| 14 Isolate* isolate, SnapshotByteSink* sink, |
| 15 FunctionCodeHandling function_code_handling) |
14 : Serializer(isolate, sink), | 16 : Serializer(isolate, sink), |
15 root_index_wave_front_(0), | 17 root_index_wave_front_(0), |
16 serializing_builtins_(false) { | 18 serializing_builtins_(false), |
17 // Clear the cache of objects used by the partial snapshot. After the | 19 function_code_handling_(function_code_handling) { |
18 // strong roots have been serialized we can create a partial snapshot | |
19 // which will repopulate the cache with objects needed by that partial | |
20 // snapshot. | |
21 isolate->partial_snapshot_cache()->Clear(); | |
22 InitializeCodeAddressMap(); | 20 InitializeCodeAddressMap(); |
23 } | 21 } |
24 | 22 |
25 StartupSerializer::~StartupSerializer() { | 23 StartupSerializer::~StartupSerializer() { |
26 OutputStatistics("StartupSerializer"); | 24 OutputStatistics("StartupSerializer"); |
27 } | 25 } |
28 | 26 |
29 void StartupSerializer::SerializeObject(HeapObject* obj, HowToCode how_to_code, | 27 void StartupSerializer::SerializeObject(HeapObject* obj, HowToCode how_to_code, |
30 WhereToPoint where_to_point, int skip) { | 28 WhereToPoint where_to_point, int skip) { |
31 DCHECK(!obj->IsJSFunction()); | 29 DCHECK(!obj->IsJSFunction()); |
32 | 30 |
33 if (obj->IsCode()) { | 31 if (obj->IsCode()) { |
34 Code* code = Code::cast(obj); | 32 Code* code = Code::cast(obj); |
35 // If the function code is compiled (either as native code or bytecode), | 33 // If the function code is compiled (either as native code or bytecode), |
36 // replace it with lazy-compile builtin. Only exception is when we are | 34 // replace it with lazy-compile builtin. Only exception is when we are |
37 // serializing the canonical interpreter-entry-trampoline builtin. | 35 // serializing the canonical interpreter-entry-trampoline builtin. |
38 if (code->kind() == Code::FUNCTION || | 36 if (function_code_handling_ == CLEAR_FUNCTION_CODE && |
39 (!serializing_builtins_ && code->is_interpreter_entry_trampoline())) { | 37 (code->kind() == Code::FUNCTION || |
| 38 (!serializing_builtins_ && code->is_interpreter_entry_trampoline()))) { |
40 obj = isolate()->builtins()->builtin(Builtins::kCompileLazy); | 39 obj = isolate()->builtins()->builtin(Builtins::kCompileLazy); |
41 } | 40 } |
42 } else if (obj->IsBytecodeArray()) { | 41 } else if (obj->IsBytecodeArray()) { |
43 obj = isolate()->heap()->undefined_value(); | 42 obj = isolate()->heap()->undefined_value(); |
44 } | 43 } |
45 | 44 |
46 int root_index = root_index_map_.Lookup(obj); | 45 int root_index = root_index_map_.Lookup(obj); |
47 bool is_immortal_immovable_root = false; | 46 bool is_immortal_immovable_root = false; |
48 // We can only encode roots as such if it has already been serialized. | 47 // We can only encode roots as such if it has already been serialized. |
49 // That applies to root indices below the wave front. | 48 // That applies to root indices below the wave front. |
(...skipping 18 matching lines...) Expand all Loading... |
68 if (is_immortal_immovable_root) { | 67 if (is_immortal_immovable_root) { |
69 // Make sure that the immortal immovable root has been included in the first | 68 // Make sure that the immortal immovable root has been included in the first |
70 // chunk of its reserved space , so that it is deserialized onto the first | 69 // chunk of its reserved space , so that it is deserialized onto the first |
71 // page of its space and stays immortal immovable. | 70 // page of its space and stays immortal immovable. |
72 BackReference ref = back_reference_map_.Lookup(obj); | 71 BackReference ref = back_reference_map_.Lookup(obj); |
73 CHECK(ref.is_valid() && ref.chunk_index() == 0); | 72 CHECK(ref.is_valid() && ref.chunk_index() == 0); |
74 } | 73 } |
75 } | 74 } |
76 | 75 |
77 void StartupSerializer::SerializeWeakReferencesAndDeferred() { | 76 void StartupSerializer::SerializeWeakReferencesAndDeferred() { |
78 // This phase comes right after the serialization (of the snapshot). | 77 // This comes right after serialization of the partial snapshot, where we |
79 // After we have done the partial serialization the partial snapshot cache | 78 // add entries to the partial snapshot cache of the startup snapshot. Add |
80 // will contain some references needed to decode the partial snapshot. We | 79 // one entry with 'undefined' to terminate the partial snapshot cache. |
81 // add one entry with 'undefined' which is the sentinel that the deserializer | |
82 // uses to know it is done deserializing the array. | |
83 Object* undefined = isolate()->heap()->undefined_value(); | 80 Object* undefined = isolate()->heap()->undefined_value(); |
84 VisitPointer(&undefined); | 81 VisitPointer(&undefined); |
85 isolate()->heap()->IterateWeakRoots(this, VISIT_ALL); | 82 isolate()->heap()->IterateWeakRoots(this, VISIT_ALL); |
86 SerializeDeferredObjects(); | 83 SerializeDeferredObjects(); |
87 Pad(); | 84 Pad(); |
88 } | 85 } |
89 | 86 |
90 void StartupSerializer::Synchronize(VisitorSynchronization::SyncTag tag) { | 87 void StartupSerializer::Synchronize(VisitorSynchronization::SyncTag tag) { |
91 // We expect the builtins tag after builtins have been serialized. | 88 // We expect the builtins tag after builtins have been serialized. |
92 DCHECK(!serializing_builtins_ || tag == VisitorSynchronization::kBuiltins); | 89 DCHECK(!serializing_builtins_ || tag == VisitorSynchronization::kBuiltins); |
93 serializing_builtins_ = (tag == VisitorSynchronization::kHandleScope); | 90 serializing_builtins_ = (tag == VisitorSynchronization::kHandleScope); |
94 sink_->Put(kSynchronize, "Synchronize"); | 91 sink_->Put(kSynchronize, "Synchronize"); |
95 } | 92 } |
96 | 93 |
97 void StartupSerializer::SerializeStrongReferences() { | 94 void StartupSerializer::SerializeStrongReferences() { |
98 Isolate* isolate = this->isolate(); | 95 Isolate* isolate = this->isolate(); |
99 // No active threads. | 96 // No active threads. |
100 CHECK_NULL(isolate->thread_manager()->FirstThreadStateInUse()); | 97 CHECK_NULL(isolate->thread_manager()->FirstThreadStateInUse()); |
101 // No active or weak handles. | 98 // No active or weak handles. |
102 CHECK(isolate->handle_scope_implementer()->blocks()->is_empty()); | 99 CHECK(isolate->handle_scope_implementer()->blocks()->is_empty()); |
103 CHECK_EQ(0, isolate->global_handles()->NumberOfWeakHandles()); | 100 CHECK_EQ(0, isolate->global_handles()->NumberOfWeakHandles()); |
104 CHECK_EQ(0, isolate->eternal_handles()->NumberOfHandles()); | 101 CHECK_EQ(0, isolate->eternal_handles()->NumberOfHandles()); |
105 // We don't support serializing installed extensions. | 102 // We don't support serializing installed extensions. |
106 CHECK(!isolate->has_installed_extensions()); | 103 CHECK(!isolate->has_installed_extensions()); |
107 isolate->heap()->IterateSmiRoots(this); | 104 isolate->heap()->IterateSmiRoots(this); |
108 isolate->heap()->IterateStrongRoots(this, VISIT_ONLY_STRONG); | 105 isolate->heap()->IterateStrongRoots(this, |
| 106 VISIT_ONLY_STRONG_FOR_SERIALIZATION); |
109 } | 107 } |
110 | 108 |
111 void StartupSerializer::VisitPointers(Object** start, Object** end) { | 109 void StartupSerializer::VisitPointers(Object** start, Object** end) { |
112 for (Object** current = start; current < end; current++) { | 110 for (Object** current = start; current < end; current++) { |
113 if (start == isolate()->heap()->roots_array_start()) { | 111 if (start == isolate()->heap()->roots_array_start()) { |
114 root_index_wave_front_ = | 112 root_index_wave_front_ = |
115 Max(root_index_wave_front_, static_cast<intptr_t>(current - start)); | 113 Max(root_index_wave_front_, static_cast<intptr_t>(current - start)); |
116 } | 114 } |
117 if (ShouldBeSkipped(current)) { | 115 if (ShouldBeSkipped(current)) { |
118 sink_->Put(kSkip, "Skip"); | 116 sink_->Put(kSkip, "Skip"); |
119 sink_->PutInt(kPointerSize, "SkipOneWord"); | 117 sink_->PutInt(kPointerSize, "SkipOneWord"); |
120 } else if ((*current)->IsSmi()) { | 118 } else if ((*current)->IsSmi()) { |
121 sink_->Put(kOnePointerRawData, "Smi"); | 119 sink_->Put(kOnePointerRawData, "Smi"); |
122 for (int i = 0; i < kPointerSize; i++) { | 120 for (int i = 0; i < kPointerSize; i++) { |
123 sink_->Put(reinterpret_cast<byte*>(current)[i], "Byte"); | 121 sink_->Put(reinterpret_cast<byte*>(current)[i], "Byte"); |
124 } | 122 } |
125 } else { | 123 } else { |
126 SerializeObject(HeapObject::cast(*current), kPlain, kStartOfObject, 0); | 124 SerializeObject(HeapObject::cast(*current), kPlain, kStartOfObject, 0); |
127 } | 125 } |
128 } | 126 } |
129 } | 127 } |
130 | 128 |
131 } // namespace internal | 129 } // namespace internal |
132 } // namespace v8 | 130 } // namespace v8 |
OLD | NEW |