Commit 70fdac9a authored by mtrofin's avatar mtrofin Committed by Commit bot

[turbofan] Correct regalloc blocked register behavior

This corrects the case when we need to allocate a
blocked register, but the blockage happens after a
use as an instruction input, and there's no place to
split before that.

BUG=v8:5888

Review-Url: https://codereview.chromium.org/2652153005
Cr-Original-Commit-Position: refs/heads/master@{#42706}
Committed: https://chromium.googlesource.com/v8/v8/+/ca779b29a60785d962d7cd636f6b0f09e66130b3
Review-Url: https://codereview.chromium.org/2652153005
Cr-Commit-Position: refs/heads/master@{#42710}
parent a8a43270
......@@ -3278,8 +3278,32 @@ void LinearScanAllocator::AllocateBlockedReg(LiveRange* current) {
register_use->pos())) {
SpillBetween(current, current->Start(), register_use->pos());
} else {
SetLiveRangeAssignedRegister(current, reg);
SplitAndSpillIntersecting(current);
// We can't spill up to the first register use, because there is no gap
// where the fill before the register use may happen. This happens when
// there is high register pressure, we are at the beginning of an
// instruction, we are the input to that instruction, and we can't hold
// on to the register past the instruction (we likely lose due to an
// output or a temp).
// We give the `reg` register to this range, but then we need to spill
// until the next register use, if any.
LifetimePosition after_this_reg_use = register_use->pos().NextFullStart();
if (after_this_reg_use >= current->End()) {
// The range ends at this instruction, since the end is at or before
// the next gap. It should follow that there is no other use either.
DCHECK_NULL(register_use->next());
SetLiveRangeAssignedRegister(current, reg);
} else {
const UsePosition* next_reg_pos = register_use->next();
for (; next_reg_pos != nullptr; next_reg_pos = next_reg_pos->next()) {
if (next_reg_pos->type() == UsePositionType::kRequiresRegister) break;
}
SetLiveRangeAssignedRegister(current, reg);
if (next_reg_pos == nullptr) {
SpillAfter(current, after_this_reg_use);
} else {
SpillBetween(current, after_this_reg_use, next_reg_pos->pos());
}
}
}
return;
}
......
// Copyright 2017 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.
load("test/mjsunit/wasm/wasm-constants.js");
load("test/mjsunit/wasm/wasm-module-builder.js");
(function() {
var builder = new WasmModuleBuilder();
builder.addMemory(32, 32, false);
builder.addFunction("test", kSig_i_iii)
.addBodyWithEnd([
// body:
kExprI64Const, 0xb4, 0x42,
kExprI64Const, 0x7a,
kExprI64Const, 0x42,
kExprI64Const, 0x7a,
kExprI64Ior,
kExprI64Ctz,
kExprI64Ctz,
kExprI64Shl,
kExprI64Mul,
kExprI64Const, 0x41,
kExprI64Ctz,
kExprI64Ctz,
kExprI64Shl,
kExprF32SConvertI64,
kExprI64Const, 0x42,
kExprI64Const, 0x02,
kExprI64Const, 0x7a,
kExprI64Mul,
kExprI64Const, 0x42,
kExprI64Ctz,
kExprI64Shl,
kExprI64Const, 0x7a,
kExprI64Ctz,
kExprI64Shl,
kExprI64Mul,
kExprI64Const, 0x41,
kExprI64Ctz,
kExprI64Ctz,
kExprI64Shl,
kExprF32SConvertI64,
kExprUnreachable,
kExprEnd, // @65
])
.exportFunc();
var module = new WebAssembly.Module(builder.toBuffer());
})();
(function() {
var builder = new WasmModuleBuilder();
builder.addMemory(16, 32, false);
builder.addFunction("test", kSig_i_iii)
.addBodyWithEnd([
// body:
kExprI64Const, 0x42,
kExprI64Const, 0x7a,
kExprI64Ctz,
kExprI64Mul,
kExprI64Ctz,
kExprI64Const, 0x41,
kExprI64Ctz,
kExprI64Ctz,
kExprI64Shl,
kExprI64Const, 0x41,
kExprI64Ctz,
kExprI64Ctz,
kExprI64Shl,
kExprF32SConvertI64,
kExprUnreachable,
kExprEnd, // @20
])
.exportFunc();
var module = new WebAssembly.Module(builder.toBuffer());
})();
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