| Index: src/interpreter/bytecode-register-allocator.cc | 
| diff --git a/src/interpreter/bytecode-register-allocator.cc b/src/interpreter/bytecode-register-allocator.cc | 
| index 4efb612db52b7566464d58a199f15e397b820350..0a617c048acb17176df0f5693dead235dc42e271 100644 | 
| --- a/src/interpreter/bytecode-register-allocator.cc | 
| +++ b/src/interpreter/bytecode-register-allocator.cc | 
| @@ -10,17 +10,173 @@ namespace v8 { | 
| namespace internal { | 
| namespace interpreter { | 
|  | 
| +TemporaryRegisterAllocator::TemporaryRegisterAllocator(Zone* zone, | 
| +                                                       int allocation_base) | 
| +    : free_temporaries_(zone), | 
| +      allocation_base_(allocation_base), | 
| +      allocation_count_(0) {} | 
| + | 
| +Register TemporaryRegisterAllocator::first_temporary_register() const { | 
| +  DCHECK(allocation_count() > 0); | 
| +  return Register(allocation_base()); | 
| +} | 
| + | 
| +Register TemporaryRegisterAllocator::last_temporary_register() const { | 
| +  DCHECK(allocation_count() > 0); | 
| +  return Register(allocation_base() + allocation_count() - 1); | 
| +} | 
| + | 
| +int TemporaryRegisterAllocator::AllocateTemporaryRegister() { | 
| +  allocation_count_ += 1; | 
| +  return allocation_base() + allocation_count() - 1; | 
| +} | 
| + | 
| +int TemporaryRegisterAllocator::BorrowTemporaryRegister() { | 
| +  if (free_temporaries_.empty()) { | 
| +    return AllocateTemporaryRegister(); | 
| +  } else { | 
| +    auto pos = free_temporaries_.begin(); | 
| +    int retval = *pos; | 
| +    free_temporaries_.erase(pos); | 
| +    return retval; | 
| +  } | 
| +} | 
| + | 
| +int TemporaryRegisterAllocator::BorrowTemporaryRegisterNotInRange( | 
| +    int start_index, int end_index) { | 
| +  if (free_temporaries_.empty()) { | 
| +    int next_allocation = allocation_base() + allocation_count(); | 
| +    while (next_allocation >= start_index && next_allocation <= end_index) { | 
| +      free_temporaries_.insert(AllocateTemporaryRegister()); | 
| +      next_allocation += 1; | 
| +    } | 
| +    return AllocateTemporaryRegister(); | 
| +  } | 
| + | 
| +  ZoneSet<int>::iterator index = free_temporaries_.lower_bound(start_index); | 
| +  if (index == free_temporaries_.begin()) { | 
| +    // If start_index is the first free register, check for a register | 
| +    // greater than end_index. | 
| +    index = free_temporaries_.upper_bound(end_index); | 
| +    if (index == free_temporaries_.end()) { | 
| +      return AllocateTemporaryRegister(); | 
| +    } | 
| +  } else { | 
| +    // If there is a free register < start_index | 
| +    index--; | 
| +  } | 
| + | 
| +  int retval = *index; | 
| +  free_temporaries_.erase(index); | 
| +  return retval; | 
| +} | 
| + | 
| +int TemporaryRegisterAllocator::PrepareForConsecutiveTemporaryRegisters( | 
| +    size_t count) { | 
| +  if (count == 0) { | 
| +    return -1; | 
| +  } | 
| + | 
| +  // TODO(oth): replace use of set<> here for free_temporaries with a | 
| +  // more efficient structure. And/or partition into two searches - | 
| +  // one before the translation window and one after. | 
| + | 
| +  // A run will require at least |count| free temporaries. | 
| +  while (free_temporaries_.size() < count) { | 
| +    free_temporaries_.insert(AllocateTemporaryRegister()); | 
| +  } | 
| + | 
| +  // Search within existing temporaries for a run. | 
| +  auto start = free_temporaries_.begin(); | 
| +  size_t run_length = 0; | 
| +  for (auto run_end = start; run_end != free_temporaries_.end(); run_end++) { | 
| +    int expected = *start + static_cast<int>(run_length); | 
| +    if (*run_end != expected) { | 
| +      start = run_end; | 
| +      run_length = 0; | 
| +    } | 
| +    Register reg_start(*start); | 
| +    Register reg_expected(expected); | 
| +    if (RegisterTranslator::DistanceToTranslationWindow(reg_start) > 0 && | 
| +        RegisterTranslator::DistanceToTranslationWindow(reg_expected) <= 0) { | 
| +      // Run straddles the lower edge of the translation window. Registers | 
| +      // after the start of this boundary are displaced by the register | 
| +      // translator to provide a hole for translation. Runs either side | 
| +      // of the boundary are fine. | 
| +      start = run_end; | 
| +      run_length = 0; | 
| +    } | 
| +    if (++run_length == count) { | 
| +      return *start; | 
| +    } | 
| +  } | 
| + | 
| +  // Continue run if possible across existing last temporary. | 
| +  if (allocation_count_ > 0 && (start == free_temporaries_.end() || | 
| +                                *start + static_cast<int>(run_length) != | 
| +                                    last_temporary_register().index() + 1)) { | 
| +    run_length = 0; | 
| +  } | 
| + | 
| +  // Pad temporaries if extended run would cross translation boundary. | 
| +  Register reg_first(*start); | 
| +  Register reg_last(*start + static_cast<int>(count) - 1); | 
| +  DCHECK_GT(RegisterTranslator::DistanceToTranslationWindow(reg_first), | 
| +            RegisterTranslator::DistanceToTranslationWindow(reg_last)); | 
| +  while (RegisterTranslator::DistanceToTranslationWindow(reg_first) > 0 && | 
| +         RegisterTranslator::DistanceToTranslationWindow(reg_last) <= 0) { | 
| +    auto pos_insert_pair = | 
| +        free_temporaries_.insert(AllocateTemporaryRegister()); | 
| +    reg_first = Register(*pos_insert_pair.first); | 
| +    reg_last = Register(reg_first.index() + static_cast<int>(count) - 1); | 
| +    run_length = 0; | 
| +  } | 
| + | 
| +  // Ensure enough registers for run. | 
| +  while (run_length++ < count) { | 
| +    free_temporaries_.insert(AllocateTemporaryRegister()); | 
| +  } | 
| + | 
| +  int run_start = | 
| +      last_temporary_register().index() - static_cast<int>(count) + 1; | 
| +  DCHECK(RegisterTranslator::DistanceToTranslationWindow(Register(run_start)) <= | 
| +             0 || | 
| +         RegisterTranslator::DistanceToTranslationWindow( | 
| +             Register(run_start + static_cast<int>(count) - 1)) > 0); | 
| +  return run_start; | 
| +} | 
| + | 
| +bool TemporaryRegisterAllocator::RegisterIsLive(Register reg) const { | 
| +  if (allocation_count_ > 0) { | 
| +    DCHECK(reg >= first_temporary_register() && | 
| +           reg <= last_temporary_register()); | 
| +    return free_temporaries_.find(reg.index()) == free_temporaries_.end(); | 
| +  } else { | 
| +    return false; | 
| +  } | 
| +} | 
| + | 
| +void TemporaryRegisterAllocator::BorrowConsecutiveTemporaryRegister( | 
| +    int reg_index) { | 
| +  DCHECK(free_temporaries_.find(reg_index) != free_temporaries_.end()); | 
| +  free_temporaries_.erase(reg_index); | 
| +} | 
| + | 
| +void TemporaryRegisterAllocator::ReturnTemporaryRegister(int reg_index) { | 
| +  DCHECK(free_temporaries_.find(reg_index) == free_temporaries_.end()); | 
| +  free_temporaries_.insert(reg_index); | 
| +} | 
| + | 
| BytecodeRegisterAllocator::BytecodeRegisterAllocator( | 
| -    BytecodeArrayBuilder* builder) | 
| -    : builder_(builder), | 
| -      allocated_(builder->zone()), | 
| +    Zone* zone, TemporaryRegisterAllocator* allocator) | 
| +    : base_allocator_(allocator), | 
| +      allocated_(zone), | 
| next_consecutive_register_(-1), | 
| next_consecutive_count_(-1) {} | 
|  | 
| - | 
| BytecodeRegisterAllocator::~BytecodeRegisterAllocator() { | 
| for (auto i = allocated_.rbegin(); i != allocated_.rend(); i++) { | 
| -    builder_->ReturnTemporaryRegister(*i); | 
| +    base_allocator()->ReturnTemporaryRegister(*i); | 
| } | 
| allocated_.clear(); | 
| } | 
| @@ -29,9 +185,9 @@ BytecodeRegisterAllocator::~BytecodeRegisterAllocator() { | 
| Register BytecodeRegisterAllocator::NewRegister() { | 
| int allocated = -1; | 
| if (next_consecutive_count_ <= 0) { | 
| -    allocated = builder_->BorrowTemporaryRegister(); | 
| +    allocated = base_allocator()->BorrowTemporaryRegister(); | 
| } else { | 
| -    allocated = builder_->BorrowTemporaryRegisterNotInRange( | 
| +    allocated = base_allocator()->BorrowTemporaryRegisterNotInRange( | 
| next_consecutive_register_, | 
| next_consecutive_register_ + next_consecutive_count_ - 1); | 
| } | 
| @@ -52,7 +208,7 @@ bool BytecodeRegisterAllocator::RegisterIsAllocatedInThisScope( | 
| void BytecodeRegisterAllocator::PrepareForConsecutiveAllocations(size_t count) { | 
| if (static_cast<int>(count) > next_consecutive_count_) { | 
| next_consecutive_register_ = | 
| -        builder_->PrepareForConsecutiveTemporaryRegisters(count); | 
| +        base_allocator()->PrepareForConsecutiveTemporaryRegisters(count); | 
| next_consecutive_count_ = static_cast<int>(count); | 
| } | 
| } | 
| @@ -61,7 +217,8 @@ void BytecodeRegisterAllocator::PrepareForConsecutiveAllocations(size_t count) { | 
| Register BytecodeRegisterAllocator::NextConsecutiveRegister() { | 
| DCHECK_GE(next_consecutive_register_, 0); | 
| DCHECK_GT(next_consecutive_count_, 0); | 
| -  builder_->BorrowConsecutiveTemporaryRegister(next_consecutive_register_); | 
| +  base_allocator()->BorrowConsecutiveTemporaryRegister( | 
| +      next_consecutive_register_); | 
| allocated_.push_back(next_consecutive_register_); | 
| next_consecutive_count_--; | 
| return Register(next_consecutive_register_++); | 
|  |