Commit 5efb8462 authored by sgjesse@chromium.org's avatar sgjesse@chromium.org

ARM: Clean up literal pool generation.

Remove dead code, and generate pools less frequently.

BUG=none
TEST=none

Review URL: http://codereview.chromium.org//7108061
Patch from Martyn Capewell <m.m.capewell@googlemail.com>.

git-svn-id: http://v8.googlecode.com/svn/branches/bleeding_edge@8309 ce2b1a6d-e550-0410-aec6-3dcde31c8c00
parent cc19d1e2
This diff is collapsed.
...@@ -1158,10 +1158,6 @@ class Assembler : public AssemblerBase { ...@@ -1158,10 +1158,6 @@ class Assembler : public AssemblerBase {
DISALLOW_IMPLICIT_CONSTRUCTORS(BlockConstPoolScope); DISALLOW_IMPLICIT_CONSTRUCTORS(BlockConstPoolScope);
}; };
// Postpone the generation of the constant pool for the specified number of
// instructions.
void BlockConstPoolFor(int instructions);
// Debugging // Debugging
// Mark address of the ExitJSFrame code. // Mark address of the ExitJSFrame code.
...@@ -1221,17 +1217,17 @@ class Assembler : public AssemblerBase { ...@@ -1221,17 +1217,17 @@ class Assembler : public AssemblerBase {
static int GetCmpImmediateRawImmediate(Instr instr); static int GetCmpImmediateRawImmediate(Instr instr);
static bool IsNop(Instr instr, int type = NON_MARKING_NOP); static bool IsNop(Instr instr, int type = NON_MARKING_NOP);
// Buffer size and constant pool distance are checked together at regular
// intervals of kBufferCheckInterval emitted bytes
static const int kBufferCheckInterval = 1*KB/2;
// Constants in pools are accessed via pc relative addressing, which can // Constants in pools are accessed via pc relative addressing, which can
// reach +/-4KB thereby defining a maximum distance between the instruction // reach +/-4KB thereby defining a maximum distance between the instruction
// and the accessed constant. We satisfy this constraint by limiting the // and the accessed constant.
// distance between pools. static const int kMaxDistToPool = 4*KB;
static const int kMaxDistBetweenPools = 4*KB - 2*kBufferCheckInterval; static const int kMaxNumPendingRelocInfo = kMaxDistToPool/kInstrSize;
static const int kMaxNumPRInfo = kMaxDistBetweenPools/kInstrSize;
// Check if is time to emit a constant pool for pending reloc info entries // Postpone the generation of the constant pool for the specified number of
// instructions.
void BlockConstPoolFor(int instructions);
// Check if is time to emit a constant pool.
void CheckConstPool(bool force_emit, bool require_jump); void CheckConstPool(bool force_emit, bool require_jump);
protected: protected:
...@@ -1256,18 +1252,37 @@ class Assembler : public AssemblerBase { ...@@ -1256,18 +1252,37 @@ class Assembler : public AssemblerBase {
// Patch branch instruction at pos to branch to given branch target pos // Patch branch instruction at pos to branch to given branch target pos
void target_at_put(int pos, int target_pos); void target_at_put(int pos, int target_pos);
// Block the emission of the constant pool before pc_offset // Prevent contant pool emission until EndBlockConstPool is called.
void BlockConstPoolBefore(int pc_offset) { // Call to this function can be nested but must be followed by an equal
if (no_const_pool_before_ < pc_offset) no_const_pool_before_ = pc_offset; // number of call to EndBlockConstpool.
}
void StartBlockConstPool() { void StartBlockConstPool() {
const_pool_blocked_nesting_++; if (const_pool_blocked_nesting_++ == 0) {
// Prevent constant pool checks happening by setting the next check to
// the biggest possible offset.
next_buffer_check_ = kMaxInt;
}
} }
// Resume constant pool emission. Need to be called as many time as
// StartBlockConstPool to have an effect.
void EndBlockConstPool() { void EndBlockConstPool() {
const_pool_blocked_nesting_--; if (--const_pool_blocked_nesting_ == 0) {
// Check the constant pool hasn't been blocked for too long.
ASSERT((num_pending_reloc_info_ == 0) ||
(pc_offset() < (first_const_pool_use_ + kMaxDistToPool)));
// Two cases:
// * no_const_pool_before_ >= next_buffer_check_ and the emission is
// still blocked
// * no_const_pool_before_ < next_buffer_check_ and the next emit will
// trigger a check.
next_buffer_check_ = no_const_pool_before_;
}
}
bool is_const_pool_blocked() const {
return (const_pool_blocked_nesting_ > 0) ||
(pc_offset() < no_const_pool_before_);
} }
bool is_const_pool_blocked() const { return const_pool_blocked_nesting_ > 0; }
private: private:
// Code buffer: // Code buffer:
...@@ -1301,33 +1316,41 @@ class Assembler : public AssemblerBase { ...@@ -1301,33 +1316,41 @@ class Assembler : public AssemblerBase {
// expensive. By default we only check again once a number of instructions // expensive. By default we only check again once a number of instructions
// has been generated. That also means that the sizing of the buffers is not // has been generated. That also means that the sizing of the buffers is not
// an exact science, and that we rely on some slop to not overrun buffers. // an exact science, and that we rely on some slop to not overrun buffers.
static const int kCheckConstIntervalInst = 32; static const int kCheckPoolIntervalInst = 32;
static const int kCheckConstInterval = kCheckConstIntervalInst * kInstrSize; static const int kCheckPoolInterval = kCheckPoolIntervalInst * kInstrSize;
// Pools are emitted after function return and in dead code at (more or less) // Average distance beetween a constant pool and the first instruction
// regular intervals of kDistBetweenPools bytes // accessing the constant pool. Longer distance should result in less I-cache
static const int kDistBetweenPools = 1*KB; // pollution.
// In practice the distance will be smaller since constant pool emission is
// forced after function return and sometimes after unconditional branches.
static const int kAvgDistToPool = kMaxDistToPool - kCheckPoolInterval;
// Emission of the constant pool may be blocked in some code sequences. // Emission of the constant pool may be blocked in some code sequences.
int const_pool_blocked_nesting_; // Block emission if this is not zero. int const_pool_blocked_nesting_; // Block emission if this is not zero.
int no_const_pool_before_; // Block emission before this pc offset. int no_const_pool_before_; // Block emission before this pc offset.
// Keep track of the last emitted pool to guarantee a maximal distance // Keep track of the first instruction requiring a constant pool entry
int last_const_pool_end_; // pc offset following the last constant pool // since the previous constant pool was emitted.
int first_const_pool_use_;
// Relocation info generation // Relocation info generation
// Each relocation is encoded as a variable size value // Each relocation is encoded as a variable size value
static const int kMaxRelocSize = RelocInfoWriter::kMaxSize; static const int kMaxRelocSize = RelocInfoWriter::kMaxSize;
RelocInfoWriter reloc_info_writer; RelocInfoWriter reloc_info_writer;
// Relocation info records are also used during code generation as temporary // Relocation info records are also used during code generation as temporary
// containers for constants and code target addresses until they are emitted // containers for constants and code target addresses until they are emitted
// to the constant pool. These pending relocation info records are temporarily // to the constant pool. These pending relocation info records are temporarily
// stored in a separate buffer until a constant pool is emitted. // stored in a separate buffer until a constant pool is emitted.
// If every instruction in a long sequence is accessing the pool, we need one // If every instruction in a long sequence is accessing the pool, we need one
// pending relocation entry per instruction. // pending relocation entry per instruction.
RelocInfo prinfo_[kMaxNumPRInfo]; // the buffer of pending relocation info
int num_prinfo_; // number of pending reloc info entries in the buffer // the buffer of pending relocation info
RelocInfo pending_reloc_info_[kMaxNumPendingRelocInfo];
// number of pending reloc info entries in the buffer
int num_pending_reloc_info_;
// The bound position, before this we cannot do instruction elimination. // The bound position, before this we cannot do instruction elimination.
int last_bound_pos_; int last_bound_pos_;
......
...@@ -445,20 +445,6 @@ void MacroAssembler::Usat(Register dst, int satpos, const Operand& src, ...@@ -445,20 +445,6 @@ void MacroAssembler::Usat(Register dst, int satpos, const Operand& src,
} }
void MacroAssembler::SmiJumpTable(Register index, Vector<Label*> targets) {
// Empty the const pool.
CheckConstPool(true, true);
add(pc, pc, Operand(index,
LSL,
Instruction::kInstrSizeLog2 - kSmiTagSize));
BlockConstPoolBefore(pc_offset() + (targets.length() + 1) * kInstrSize);
nop(); // Jump table alignment.
for (int i = 0; i < targets.length(); i++) {
b(targets[i]);
}
}
void MacroAssembler::LoadRoot(Register destination, void MacroAssembler::LoadRoot(Register destination,
Heap::RootListIndex index, Heap::RootListIndex index,
Condition cond) { Condition cond) {
......
...@@ -146,8 +146,6 @@ class MacroAssembler: public Assembler { ...@@ -146,8 +146,6 @@ class MacroAssembler: public Assembler {
void Move(Register dst, Register src); void Move(Register dst, Register src);
void Move(DoubleRegister dst, DoubleRegister src); void Move(DoubleRegister dst, DoubleRegister src);
// Jumps to the label at the index given by the Smi in "index".
void SmiJumpTable(Register index, Vector<Label*> targets);
// Load an object from the root table. // Load an object from the root table.
void LoadRoot(Register destination, void LoadRoot(Register destination,
Heap::RootListIndex index, Heap::RootListIndex index,
......
...@@ -899,13 +899,12 @@ void RegExpMacroAssemblerARM::PushBacktrack(Label* label) { ...@@ -899,13 +899,12 @@ void RegExpMacroAssemblerARM::PushBacktrack(Label* label) {
constant_offset - offset_of_pc_register_read; constant_offset - offset_of_pc_register_read;
ASSERT(pc_offset_of_constant < 0); ASSERT(pc_offset_of_constant < 0);
if (is_valid_memory_offset(pc_offset_of_constant)) { if (is_valid_memory_offset(pc_offset_of_constant)) {
masm_->BlockConstPoolBefore(masm_->pc_offset() + Assembler::kInstrSize); Assembler::BlockConstPoolScope block_const_pool(masm_);
__ ldr(r0, MemOperand(pc, pc_offset_of_constant)); __ ldr(r0, MemOperand(pc, pc_offset_of_constant));
} else { } else {
// Not a 12-bit offset, so it needs to be loaded from the constant // Not a 12-bit offset, so it needs to be loaded from the constant
// pool. // pool.
masm_->BlockConstPoolBefore( Assembler::BlockConstPoolScope block_const_pool(masm_);
masm_->pc_offset() + 2 * Assembler::kInstrSize);
__ mov(r0, Operand(pc_offset_of_constant + Assembler::kInstrSize)); __ mov(r0, Operand(pc_offset_of_constant + Assembler::kInstrSize));
__ ldr(r0, MemOperand(pc, r0)); __ ldr(r0, MemOperand(pc, r0));
} }
...@@ -1185,8 +1184,7 @@ void RegExpMacroAssemblerARM::CheckStackLimit() { ...@@ -1185,8 +1184,7 @@ void RegExpMacroAssemblerARM::CheckStackLimit() {
void RegExpMacroAssemblerARM::EmitBacktrackConstantPool() { void RegExpMacroAssemblerARM::EmitBacktrackConstantPool() {
__ CheckConstPool(false, false); __ CheckConstPool(false, false);
__ BlockConstPoolBefore( Assembler::BlockConstPoolScope block_const_pool(masm_);
masm_->pc_offset() + kBacktrackConstantPoolSize * Assembler::kInstrSize);
backtrack_constant_pool_offset_ = masm_->pc_offset(); backtrack_constant_pool_offset_ = masm_->pc_offset();
for (int i = 0; i < kBacktrackConstantPoolSize; i++) { for (int i = 0; i < kBacktrackConstantPoolSize; i++) {
__ emit(0); __ emit(0);
......
Markdown is supported
0% or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment