// Copyright 2015 the V8 project authors. All rights reserved.
|
// Use of this source code is governed by a BSD-style license that can be
|
// found in the LICENSE file.
|
|
#ifndef V8_INTERPRETER_BYTECODE_ARRAY_WRITER_H_
|
#define V8_INTERPRETER_BYTECODE_ARRAY_WRITER_H_
|
|
#include "src/base/compiler-specific.h"
|
#include "src/globals.h"
|
#include "src/interpreter/bytecodes.h"
|
#include "src/source-position-table.h"
|
|
namespace v8 {
|
namespace internal {
|
|
class BytecodeArray;
|
class SourcePositionTableBuilder;
|
|
namespace interpreter {
|
|
class BytecodeLabel;
|
class BytecodeNode;
|
class BytecodeJumpTable;
|
class ConstantArrayBuilder;
|
|
namespace bytecode_array_writer_unittest {
|
class BytecodeArrayWriterUnittest;
|
} // namespace bytecode_array_writer_unittest
|
|
// Class for emitting bytecode as the final stage of the bytecode
|
// generation pipeline.
|
class V8_EXPORT_PRIVATE BytecodeArrayWriter final {
|
public:
|
BytecodeArrayWriter(
|
Zone* zone, ConstantArrayBuilder* constant_array_builder,
|
SourcePositionTableBuilder::RecordingMode source_position_mode);
|
|
void Write(BytecodeNode* node);
|
void WriteJump(BytecodeNode* node, BytecodeLabel* label);
|
void WriteSwitch(BytecodeNode* node, BytecodeJumpTable* jump_table);
|
void BindLabel(BytecodeLabel* label);
|
void BindLabel(const BytecodeLabel& target, BytecodeLabel* label);
|
void BindJumpTableEntry(BytecodeJumpTable* jump_table, int case_value);
|
Handle<BytecodeArray> ToBytecodeArray(Isolate* isolate, int register_count,
|
int parameter_count,
|
Handle<ByteArray> handler_table);
|
|
private:
|
// Maximum sized packed bytecode is comprised of a prefix bytecode,
|
// plus the actual bytecode, plus the maximum number of operands times
|
// the maximum operand size.
|
static const size_t kMaxSizeOfPackedBytecode =
|
2 * sizeof(Bytecode) +
|
Bytecodes::kMaxOperands * static_cast<size_t>(OperandSize::kLast);
|
|
// Constants that act as placeholders for jump operands to be
|
// patched. These have operand sizes that match the sizes of
|
// reserved constant pool entries.
|
const uint32_t k8BitJumpPlaceholder = 0x7f;
|
const uint32_t k16BitJumpPlaceholder =
|
k8BitJumpPlaceholder | (k8BitJumpPlaceholder << 8);
|
const uint32_t k32BitJumpPlaceholder =
|
k16BitJumpPlaceholder | (k16BitJumpPlaceholder << 16);
|
|
void PatchJump(size_t jump_target, size_t jump_location);
|
void PatchJumpWith8BitOperand(size_t jump_location, int delta);
|
void PatchJumpWith16BitOperand(size_t jump_location, int delta);
|
void PatchJumpWith32BitOperand(size_t jump_location, int delta);
|
|
void EmitBytecode(const BytecodeNode* const node);
|
void EmitJump(BytecodeNode* node, BytecodeLabel* label);
|
void EmitSwitch(BytecodeNode* node, BytecodeJumpTable* jump_table);
|
void UpdateSourcePositionTable(const BytecodeNode* const node);
|
|
void UpdateExitSeenInBlock(Bytecode bytecode);
|
|
void MaybeElideLastBytecode(Bytecode next_bytecode, bool has_source_info);
|
void InvalidateLastBytecode();
|
|
ZoneVector<uint8_t>* bytecodes() { return &bytecodes_; }
|
SourcePositionTableBuilder* source_position_table_builder() {
|
return &source_position_table_builder_;
|
}
|
ConstantArrayBuilder* constant_array_builder() {
|
return constant_array_builder_;
|
}
|
|
ZoneVector<uint8_t> bytecodes_;
|
int unbound_jumps_;
|
SourcePositionTableBuilder source_position_table_builder_;
|
ConstantArrayBuilder* constant_array_builder_;
|
|
Bytecode last_bytecode_;
|
size_t last_bytecode_offset_;
|
bool last_bytecode_had_source_info_;
|
bool elide_noneffectful_bytecodes_;
|
|
bool exit_seen_in_block_;
|
|
friend class bytecode_array_writer_unittest::BytecodeArrayWriterUnittest;
|
DISALLOW_COPY_AND_ASSIGN(BytecodeArrayWriter);
|
};
|
|
} // namespace interpreter
|
} // namespace internal
|
} // namespace v8
|
|
#endif // V8_INTERPRETER_BYTECODE_ARRAY_WRITER_H_
|