e639eafea3
This is a reland of352b9ecbdb
The test/fix CL has been merged in, as the fixes to return slot accounting are needed to fix Arm64 issues turned up by the fuzzers: https://chromium-review.googlesource.com/c/v8/v8/+/2644139 The reverted fix for Wasm return slot allocation is added in patchset #2, to avoid fuzzer issues that it fixed: https://chromium-review.googlesource.com/c/v8/v8/+/2683024 TBR=neis@chromium.org Original change's description: > Reland "Reland "[compiler][wasm] Align Frame slots to value size"" > > This is a reland of1694925c72
> > Minor fix to linkage for constexpr. > > TBR=ahaas@chromium.org,neis@chromium.org > > Original change's description: > > Reland "[compiler][wasm] Align Frame slots to value size" > > > > This is a reland ofcddaf66c37
> > > > Original change's description: > > > [compiler][wasm] Align Frame slots to value size > > > > > > - Adds an AlignedSlotAllocator class and tests, to unify slot > > > allocation. This attempts to use alignment holes for smaller > > > values. > > > - Reworks Frame to use the new allocator for stack slots. > > > - Reworks LinkageAllocator to use the new allocator for stack > > > slots and for ARMv7 FP register aliasing. > > > - Fixes the RegisterAllocator to align spill slots. > > > - Fixes InstructionSelector to align spill slots. > > > > > > Bug: v8:9198 > > > > > > Change-Id: Ida148db428be89ef95de748ec5fc0e7b0358f523 > > > Reviewed-on: https://chromium-review.googlesource.com/c/v8/v8/+/2512840 > > > Commit-Queue: Bill Budge <bbudge@chromium.org> > > > Reviewed-by: Georg Neis <neis@chromium.org> > > > Reviewed-by: Andreas Haas <ahaas@chromium.org> > > > Cr-Commit-Position: refs/heads/master@{#71644} > > > > Bug: v8:9198 > > Change-Id: Ib91fa6746370c38496706341e12d05c7bf999389 > > Reviewed-on: https://chromium-review.googlesource.com/c/v8/v8/+/2633390 > > Commit-Queue: Bill Budge <bbudge@chromium.org> > > Reviewed-by: Andreas Haas <ahaas@chromium.org> > > Reviewed-by: Georg Neis <neis@chromium.org> > > Cr-Commit-Position: refs/heads/master@{#72195} > > Bug: v8:9198 > Change-Id: I91e02b823af8ec925dacf075388fb22e3eeb3384 > Reviewed-on: https://chromium-review.googlesource.com/c/v8/v8/+/2640890 > Reviewed-by: Bill Budge <bbudge@chromium.org> > Commit-Queue: Bill Budge <bbudge@chromium.org> > Cr-Commit-Position: refs/heads/master@{#72209} Bug: v8:9198 Change-Id: Ia5cf63af4e5991bc7cf42da9972ffd044fc829f0 Reviewed-on: https://chromium-review.googlesource.com/c/v8/v8/+/2733177 Commit-Queue: Bill Budge <bbudge@chromium.org> Reviewed-by: Andreas Haas <ahaas@chromium.org> Cr-Commit-Position: refs/heads/master@{#73238}
112 lines
4.4 KiB
C++
112 lines
4.4 KiB
C++
// Copyright 2018 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.
|
|
|
|
#include "src/compiler/wasm-compiler.h"
|
|
|
|
#include "src/codegen/machine-type.h"
|
|
#include "src/codegen/signature.h"
|
|
#include "src/compiler/linkage.h"
|
|
#include "src/wasm/value-type.h"
|
|
#include "src/wasm/wasm-linkage.h"
|
|
#include "test/unittests/test-utils.h"
|
|
|
|
namespace v8 {
|
|
namespace internal {
|
|
namespace wasm {
|
|
|
|
class WasmCallDescriptorTest : public TestWithZone {};
|
|
|
|
TEST_F(WasmCallDescriptorTest, TestExternRefIsGrouped) {
|
|
constexpr size_t kMaxCount = 30;
|
|
ValueType params[kMaxCount];
|
|
|
|
for (size_t i = 0; i < kMaxCount; i += 2) {
|
|
params[i] = kWasmExternRef;
|
|
EXPECT_TRUE(i + 1 < kMaxCount);
|
|
params[i + 1] = kWasmI32;
|
|
}
|
|
|
|
for (size_t count = 1; count <= kMaxCount; ++count) {
|
|
FunctionSig sig(/*return_count=*/0, count, params);
|
|
compiler::CallDescriptor* desc =
|
|
compiler::GetWasmCallDescriptor(zone(), &sig);
|
|
|
|
// The WasmInstance is the implicit first parameter.
|
|
EXPECT_EQ(count + 1, desc->ParameterCount());
|
|
|
|
bool has_untagged_stack_param = false;
|
|
bool has_tagged_register_param = false;
|
|
int max_tagged_stack_location = std::numeric_limits<int>::min();
|
|
int min_untagged_stack_location = std::numeric_limits<int>::max();
|
|
for (size_t i = 1; i < desc->ParameterCount(); ++i) {
|
|
// InputLocation i + 1, because target is the first input.
|
|
compiler::LinkageLocation location = desc->GetInputLocation(i + 1);
|
|
if (desc->GetParameterType(i).IsTagged()) {
|
|
if (location.IsRegister()) {
|
|
has_tagged_register_param = true;
|
|
} else {
|
|
EXPECT_TRUE(location.IsCallerFrameSlot());
|
|
max_tagged_stack_location =
|
|
std::max(max_tagged_stack_location, location.AsCallerFrameSlot());
|
|
}
|
|
} else { // !isTagged()
|
|
if (location.IsCallerFrameSlot()) {
|
|
has_untagged_stack_param = true;
|
|
min_untagged_stack_location = std::min(min_untagged_stack_location,
|
|
location.AsCallerFrameSlot());
|
|
} else {
|
|
EXPECT_TRUE(location.IsRegister());
|
|
}
|
|
}
|
|
}
|
|
// There should never be a tagged parameter in a register and an untagged
|
|
// parameter on the stack at the same time.
|
|
EXPECT_EQ(false, has_tagged_register_param && has_untagged_stack_param);
|
|
EXPECT_TRUE(max_tagged_stack_location < min_untagged_stack_location);
|
|
}
|
|
}
|
|
|
|
TEST_F(WasmCallDescriptorTest, Regress_1174500) {
|
|
// Our test signature should have just enough params and returns to force
|
|
// 1 param and 1 return to be allocated as stack slots. Use FP registers to
|
|
// avoid interference with implicit parameters, like the Wasm Instance.
|
|
constexpr int kParamRegisters = arraysize(kFpParamRegisters);
|
|
constexpr int kParams = kParamRegisters + 1;
|
|
constexpr int kReturnRegisters = arraysize(kFpReturnRegisters);
|
|
constexpr int kReturns = kReturnRegisters + 1;
|
|
ValueType types[kReturns + kParams];
|
|
// One S128 return slot which shouldn't be padded unless the arguments area
|
|
// of the frame requires it.
|
|
for (int i = 0; i < kReturnRegisters; ++i) types[i] = kWasmF32;
|
|
types[kReturnRegisters] = kWasmS128;
|
|
// One F32 parameter slot to misalign the parameter area.
|
|
for (int i = 0; i < kParamRegisters; ++i) types[kReturns + i] = kWasmF32;
|
|
types[kReturns + kParamRegisters] = kWasmF32;
|
|
|
|
FunctionSig sig(kReturns, kParams, types);
|
|
compiler::CallDescriptor* desc =
|
|
compiler::GetWasmCallDescriptor(zone(), &sig);
|
|
|
|
// Get the location of our stack parameter slot. Skip the implicit Wasm
|
|
// instance parameter.
|
|
compiler::LinkageLocation last_param = desc->GetInputLocation(kParams + 1);
|
|
EXPECT_TRUE(last_param.IsCallerFrameSlot());
|
|
EXPECT_EQ(MachineType::Float32(), last_param.GetType());
|
|
EXPECT_EQ(-1, last_param.GetLocation());
|
|
|
|
// The stack return slot should be right above our last parameter, and any
|
|
// argument padding slots. The return slot itself should not be padded.
|
|
const int padding = kPadArguments ? 1 : 0;
|
|
const int first_return_slot = -1 - (padding + 1);
|
|
compiler::LinkageLocation return_location =
|
|
desc->GetReturnLocation(kReturns - 1);
|
|
EXPECT_TRUE(return_location.IsCallerFrameSlot());
|
|
EXPECT_EQ(MachineType::Simd128(), return_location.GetType());
|
|
EXPECT_EQ(first_return_slot, return_location.GetLocation());
|
|
}
|
|
|
|
} // namespace wasm
|
|
} // namespace internal
|
|
} // namespace v8
|