[compiler] Fix another SIMD register spilling issue
Spilling was already fixed if a fixed SIMD register overlaps with an allocated FP register, but the other way around was missing: If an odd FP register (in this case d1) is used as a fixed output register, but this register is in use as the upper half of a SIMD register (in this case q0), we did not detect this and would just use overwrite the SIMD half. This CL also fixes this case. R=thibaudm@chromium.org Bug: v8:12330, chromium:1284980 Change-Id: Id3f98b7accd77e38ab4cd5ff8910aaf5ad96a1ed Reviewed-on: https://chromium-review.googlesource.com/c/v8/v8/+/3372910 Reviewed-by: Thibaud Michaud <thibaudm@chromium.org> Commit-Queue: Clemens Backes <clemensb@chromium.org> Cr-Commit-Position: refs/heads/main@{#78514}
This commit is contained in:
parent
3170b9903d
commit
1b3d232321
@ -2569,7 +2569,21 @@ void SinglePassRegisterAllocator::ReserveFixedRegister(
|
|||||||
// Also potentially spill the "sibling SIMD register" on architectures where a
|
// Also potentially spill the "sibling SIMD register" on architectures where a
|
||||||
// SIMD register aliases two FP registers.
|
// SIMD register aliases two FP registers.
|
||||||
if (!kSimpleFPAliasing && rep == MachineRepresentation::kSimd128) {
|
if (!kSimpleFPAliasing && rep == MachineRepresentation::kSimd128) {
|
||||||
if (!IsFreeOrSameVirtualRegister(reg.simdSibling(), virtual_register) &&
|
if (register_state()->IsAllocated(reg.simdSibling()) &&
|
||||||
|
!DefinedAfter(virtual_register, instr_index, pos)) {
|
||||||
|
SpillRegister(reg.simdSibling());
|
||||||
|
}
|
||||||
|
}
|
||||||
|
// Similarly (but the other way around), spill a SIMD register that (partly)
|
||||||
|
// overlaps with a fixed FP register. If {reg} is the lower half (i.e. an even
|
||||||
|
// register index), this is already checked above, so we only check if {reg}
|
||||||
|
// is the upper half.
|
||||||
|
if (!kSimpleFPAliasing && (reg.ToInt() & 1)) {
|
||||||
|
DCHECK_NE(MachineRepresentation::kSimd128, rep);
|
||||||
|
int allocated_vreg = VirtualRegisterForRegister(reg.simdSibling());
|
||||||
|
if (allocated_vreg != InstructionOperand::kInvalidVirtualRegister &&
|
||||||
|
VirtualRegisterDataFor(allocated_vreg).rep() ==
|
||||||
|
MachineRepresentation::kSimd128 &&
|
||||||
!DefinedAfter(virtual_register, instr_index, pos)) {
|
!DefinedAfter(virtual_register, instr_index, pos)) {
|
||||||
SpillRegister(reg.simdSibling());
|
SpillRegister(reg.simdSibling());
|
||||||
}
|
}
|
||||||
|
38
test/mjsunit/regress/wasm/regress-1284980.js
Normal file
38
test/mjsunit/regress/wasm/regress-1284980.js
Normal file
@ -0,0 +1,38 @@
|
|||||||
|
// Copyright 2022 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.
|
||||||
|
|
||||||
|
// Flags: --no-liftoff --turbo-force-mid-tier-regalloc
|
||||||
|
|
||||||
|
d8.file.execute('test/mjsunit/wasm/wasm-module-builder.js');
|
||||||
|
|
||||||
|
const builder = new WasmModuleBuilder();
|
||||||
|
builder.addMemory(16, 32, false);
|
||||||
|
builder.addType(makeSig([kWasmI32, kWasmI32, kWasmI32], []));
|
||||||
|
builder.addType(makeSig([kWasmS128], []));
|
||||||
|
builder.addType(makeSig([], [kWasmF64, kWasmF64]));
|
||||||
|
builder.addTag(makeSig([], []));
|
||||||
|
// Generate function 1 (out of 3).
|
||||||
|
builder.addFunction(undefined, 0 /* sig */)
|
||||||
|
.addLocals(kWasmI32, 2).addLocals(kWasmF32, 1).addLocals(kWasmI32, 1).addLocals(kWasmF64, 1)
|
||||||
|
.addBody([
|
||||||
|
kExprTry, kWasmVoid, // try i32
|
||||||
|
kExprCallFunction, 2, // call function #2
|
||||||
|
kExprI32Const, 0, // i32.const
|
||||||
|
kExprSelect, // select
|
||||||
|
kExprI32SConvertF64, // i32.trunc_f64_s
|
||||||
|
kExprI32Const, 0x00, // i32.const
|
||||||
|
kSimdPrefix, kExprI8x16Splat, // i8x16.splat
|
||||||
|
kSimdPrefix, kExprS128Store8Lane, 0x00, 0x00, 0x00, // s128.store8_lane
|
||||||
|
kExprCatch, 0, // catch
|
||||||
|
kExprCatchAll, // catch-all
|
||||||
|
kExprEnd, // end
|
||||||
|
kExprI32Const, 0x00, // i32.const
|
||||||
|
kSimdPrefix, kExprI8x16Splat, // i8x16.splat
|
||||||
|
kExprCallFunction, 1, // call function #1
|
||||||
|
]);
|
||||||
|
// Generate function 2 (out of 3).
|
||||||
|
builder.addFunction(undefined, 1 /* sig */).addBody([kExprUnreachable]);
|
||||||
|
// Generate function 3 (out of 3).
|
||||||
|
builder.addFunction(undefined, 2 /* sig */).addBody([kExprUnreachable]);
|
||||||
|
builder.toModule();
|
Loading…
Reference in New Issue
Block a user