OLD | NEW |
---|---|
(Empty) | |
1 // Copyright 2015 the V8 project authors. All rights reserved. | |
2 // Use of this source code is governed by a BSD-style license that can be | |
3 // found in the LICENSE file. | |
4 | |
5 #ifndef V8_INTERPRETER_BYTECODE_WRITER_H_ | |
6 #define V8_INTERPRETER_BYTECODE_WRITER_H_ | |
7 | |
8 #include "src/interpreter/bytecode-register-allocator.h" | |
9 #include "src/interpreter/bytecodes.h" | |
10 #include "src/zone-containers.h" | |
11 | |
12 namespace v8 { | |
13 namespace internal { | |
14 namespace interpreter { | |
15 | |
16 class BytecodeArrayBuilder; | |
17 class SourcePositionTableBuilder; | |
18 class TemporaryRegisterAllocator; | |
19 | |
20 class BytecodeNodeAllocator; | |
21 | |
22 // Source code position information. | |
23 class BytecodeSourceInfo final { | |
rmcilroy
2016/05/06 15:25:48
Could we just use PositionTableEntry from source-p
oth
2016/05/09 11:02:17
Different layouts and sizes. Different semantics.
| |
24 public: | |
25 static const int kUninitializedPosition = -1; | |
26 | |
27 BytecodeSourceInfo(int position = kUninitializedPosition, | |
28 bool is_statement = false) | |
29 : source_position_(position), is_statement_(is_statement) {} | |
30 | |
31 // Combine later source info with current. | |
32 void Update(const BytecodeSourceInfo& entry); | |
33 | |
34 int source_position() const { | |
35 DCHECK(is_valid()); | |
36 return source_position_; | |
37 } | |
38 | |
39 bool is_statement() const { return is_valid() && is_statement_; } | |
40 | |
41 bool is_valid() const { return source_position_ != kUninitializedPosition; } | |
42 void set_invalid() { source_position_ = kUninitializedPosition; } | |
43 | |
44 bool operator==(const BytecodeSourceInfo& other) const { | |
45 return source_position_ == other.source_position_ && | |
46 is_statement_ == other.is_statement_; | |
47 } | |
48 bool operator!=(const BytecodeSourceInfo& other) const { | |
49 return source_position_ != other.source_position_ || | |
50 is_statement_ != other.is_statement_; | |
51 } | |
52 | |
53 private: | |
54 int source_position_; | |
55 bool is_statement_; | |
56 | |
57 DISALLOW_COPY_AND_ASSIGN(BytecodeSourceInfo); | |
58 }; | |
59 | |
60 // A container for a generated bytecode, it's operands, and source information. | |
61 // These must be allocated by a BytecodeNodeAllocator instance. | |
62 class BytecodeNode final : ZoneObject { | |
63 public: | |
64 // Print to stream. | |
65 void Print(std::ostream& os); | |
66 | |
67 // Return the size when this node is serialized to a bytecode array. | |
68 size_t Size() const; | |
69 | |
70 // Return to node to allocator for re-use. | |
71 void Release(); | |
72 | |
73 Bytecode bytecode() const { return bytecode_; } | |
74 void set_bytecode(Bytecode bytecode); | |
75 void set_bytecode(Bytecode bytecode, uint32_t operand0, | |
76 OperandScale operand_scale); | |
77 void set_bytecode(Bytecode bytecode, uint32_t operand0, uint32_t operand1, | |
78 OperandScale operand_scale); | |
79 void set_bytecode(Bytecode bytecode, uint32_t operand0, uint32_t operand1, | |
80 uint32_t operand2, OperandScale operand_scale); | |
81 void set_bytecode(Bytecode bytecode, uint32_t operand0, uint32_t operand1, | |
82 uint32_t operand2, uint32_t operand3, | |
83 OperandScale operand_scale); | |
84 | |
85 uint32_t* operands() { return operands_; } | |
86 const uint32_t* operands() const { return operands_; } | |
87 | |
88 int operand_count() const { return Bytecodes::NumberOfOperands(bytecode_); } | |
89 | |
90 OperandScale operand_scale() const { return operand_scale_; } | |
91 | |
92 const BytecodeSourceInfo& source_info() const { return source_info_; } | |
93 BytecodeSourceInfo& source_info() { return source_info_; } | |
94 | |
95 private: | |
96 static const int kInvalidPosition = kMinInt; | |
97 static const size_t kMaxOperands = 4; | |
98 | |
99 friend class BytecodeNodeAllocator; | |
100 explicit BytecodeNode(BytecodeNodeAllocator* allocator); | |
101 | |
102 Bytecode bytecode_; | |
103 uint32_t operands_[kMaxOperands]; | |
104 OperandScale operand_scale_; | |
105 BytecodeSourceInfo source_info_; | |
106 BytecodeNodeAllocator* allocator_; | |
107 BytecodeNode* next_; | |
108 | |
109 DISALLOW_COPY_AND_ASSIGN(BytecodeNode); | |
110 }; | |
111 | |
112 // Allocator for BytecodeNodes that zone allocates them when needed | |
113 // and maintains a freelist to reduce number of allocations and limit | |
114 // memory use. | |
115 class BytecodeNodeAllocator final { | |
rmcilroy
2016/05/06 15:25:48
I'd be really happy if we could avoid having to ha
oth
2016/05/09 11:02:17
As discussed, let's measure it and see.
| |
116 public: | |
117 explicit BytecodeNodeAllocator(Zone* zone) | |
118 : allocation_count_(0), free_list_(nullptr), zone_(zone) {} | |
119 | |
120 BytecodeNode* Allocate(); | |
121 | |
122 private: | |
123 friend class BytecodeNode; | |
124 static const int kMaxDebugAllocations = 8; | |
125 | |
126 void Release(BytecodeNode* node); | |
127 | |
128 Zone* zone() const { return zone_; } | |
129 | |
130 int allocation_count_; | |
131 BytecodeNode* free_list_; | |
132 Zone* zone_; | |
133 | |
134 DISALLOW_COPY_AND_ASSIGN(BytecodeNodeAllocator); | |
135 }; | |
136 | |
137 // Interface for bytecode writers. | |
138 class BytecodeWriter { | |
rmcilroy
2016/05/06 15:25:48
How about calling this BytecodePipelineStage, and
oth
2016/05/09 11:02:17
Done.
| |
139 public: | |
140 virtual ~BytecodeWriter() {} | |
141 | |
142 // Flush state for bytecode array offset calculation. Returns the | |
143 // current size of bytecode array. | |
144 virtual size_t FlushForOffset() = 0; | |
145 | |
146 // Signal end of basic block. | |
147 virtual void LeaveBasicBlock() = 0; | |
148 | |
149 // Write bytecode into pipeline. The callee owns node from here and | |
150 // the caller should not hold any references to it. | |
151 virtual void Write(BytecodeNode* node) = 0; | |
152 }; | |
153 | |
154 // Class for emitting bytecode as the final stage of a bytecode writer | |
155 // chain. | |
156 class FinalStageBytecodeWriter : public BytecodeWriter { | |
rmcilroy
2016/05/06 15:25:48
Nit - could we move this to it's own file.
oth
2016/05/09 11:02:17
Done.
| |
157 public: | |
158 FinalStageBytecodeWriter( | |
159 Zone* zone, SourcePositionTableBuilder* source_position_table_builder); | |
160 virtual ~FinalStageBytecodeWriter(); | |
161 | |
162 size_t FlushForOffset() override; | |
163 void LeaveBasicBlock() override; | |
164 void Write(BytecodeNode* node) override; | |
165 | |
166 // Get the bytecode vector. | |
167 ZoneVector<uint8_t>* bytecodes() { return &bytecodes_; } | |
168 | |
169 // Returns the size in bytes of the frame associated with the | |
170 // bytecode written. | |
171 int GetMeasuredFrameSize(); | |
172 | |
173 private: | |
174 void EmitBytecode(const BytecodeNode* const node); | |
175 void UpdateSourcePositionTable(const BytecodeNode* const node); | |
176 | |
177 ZoneVector<uint8_t> bytecodes_; | |
178 int frame_register_count_; | |
179 SourcePositionTableBuilder* source_position_table_builder_; | |
180 | |
181 DISALLOW_COPY_AND_ASSIGN(FinalStageBytecodeWriter); | |
182 }; | |
183 | |
184 } // namespace interpreter | |
185 } // namespace internal | |
186 } // namespace v8 | |
187 | |
188 #endif // V8_INTERPRETER_BYTECODE_WRITER_H_ | |
OLD | NEW |