8326541: [AArch64] ZGC C2 load barrier stub should consider the length of live registers when spilling registers
Reviewed-by: eosterlund, rcastanedalo
This commit is contained in:
parent
438e64310d
commit
5c3838605d
@ -1126,6 +1126,10 @@ extern RegMask _NO_SPECIAL_REG32_mask;
|
||||
extern RegMask _NO_SPECIAL_REG_mask;
|
||||
extern RegMask _NO_SPECIAL_PTR_REG_mask;
|
||||
|
||||
// Figure out which register class each belongs in: rc_int, rc_float or
|
||||
// rc_stack.
|
||||
enum RC { rc_bad, rc_int, rc_float, rc_predicate, rc_stack };
|
||||
|
||||
class CallStubImpl {
|
||||
|
||||
//--------------------------------------------------------------
|
||||
@ -1900,10 +1904,6 @@ const Pipeline * MachEpilogNode::pipeline() const {
|
||||
|
||||
//=============================================================================
|
||||
|
||||
// Figure out which register class each belongs in: rc_int, rc_float or
|
||||
// rc_stack.
|
||||
enum RC { rc_bad, rc_int, rc_float, rc_predicate, rc_stack };
|
||||
|
||||
static enum RC rc_class(OptoReg::Name reg) {
|
||||
|
||||
if (reg == OptoReg::Bad) {
|
||||
|
@ -1,5 +1,5 @@
|
||||
/*
|
||||
* Copyright (c) 2019, 2023, Oracle and/or its affiliates. All rights reserved.
|
||||
* Copyright (c) 2019, 2024, Oracle and/or its affiliates. All rights reserved.
|
||||
* DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER.
|
||||
*
|
||||
* This code is free software; you can redistribute it and/or modify it
|
||||
@ -1081,6 +1081,35 @@ void ZBarrierSetAssembler::generate_c1_store_barrier_runtime_stub(StubAssembler*
|
||||
|
||||
#ifdef COMPILER2
|
||||
|
||||
OptoReg::Name ZBarrierSetAssembler::encode_float_vector_register_size(const Node* node, OptoReg::Name opto_reg) {
|
||||
switch (node->ideal_reg()) {
|
||||
case Op_RegF:
|
||||
// No need to refine. The original encoding is already fine to distinguish.
|
||||
assert(opto_reg % 4 == 0, "Float register should only occupy a single slot");
|
||||
break;
|
||||
// Use different encoding values of the same fp/vector register to help distinguish different sizes.
|
||||
// Such as V16. The OptoReg::name and its corresponding slot value are
|
||||
// "V16": 64, "V16_H": 65, "V16_J": 66, "V16_K": 67.
|
||||
case Op_RegD:
|
||||
case Op_VecD:
|
||||
opto_reg &= ~3;
|
||||
opto_reg |= 1;
|
||||
break;
|
||||
case Op_VecX:
|
||||
opto_reg &= ~3;
|
||||
opto_reg |= 2;
|
||||
break;
|
||||
case Op_VecA:
|
||||
opto_reg &= ~3;
|
||||
opto_reg |= 3;
|
||||
break;
|
||||
default:
|
||||
assert(false, "unexpected ideal register");
|
||||
ShouldNotReachHere();
|
||||
}
|
||||
return opto_reg;
|
||||
}
|
||||
|
||||
OptoReg::Name ZBarrierSetAssembler::refine_register(const Node* node, OptoReg::Name opto_reg) {
|
||||
if (!OptoReg::is_reg(opto_reg)) {
|
||||
return OptoReg::Bad;
|
||||
@ -1088,7 +1117,7 @@ OptoReg::Name ZBarrierSetAssembler::refine_register(const Node* node, OptoReg::N
|
||||
|
||||
const VMReg vm_reg = OptoReg::as_VMReg(opto_reg);
|
||||
if (vm_reg->is_FloatRegister()) {
|
||||
return opto_reg & ~1;
|
||||
opto_reg = encode_float_vector_register_size(node, opto_reg);
|
||||
}
|
||||
|
||||
return opto_reg;
|
||||
@ -1099,28 +1128,82 @@ OptoReg::Name ZBarrierSetAssembler::refine_register(const Node* node, OptoReg::N
|
||||
|
||||
class ZSaveLiveRegisters {
|
||||
private:
|
||||
struct RegisterData {
|
||||
VMReg _reg;
|
||||
int _slots; // slots occupied once pushed into stack
|
||||
|
||||
// Used by GrowableArray::find()
|
||||
bool operator == (const RegisterData& other) {
|
||||
return _reg == other._reg;
|
||||
}
|
||||
};
|
||||
|
||||
MacroAssembler* const _masm;
|
||||
RegSet _gp_regs;
|
||||
FloatRegSet _fp_regs;
|
||||
FloatRegSet _neon_regs;
|
||||
FloatRegSet _sve_regs;
|
||||
PRegSet _p_regs;
|
||||
|
||||
public:
|
||||
void initialize(ZBarrierStubC2* stub) {
|
||||
// Record registers that needs to be saved/restored
|
||||
int index = -1;
|
||||
GrowableArray<RegisterData> registers;
|
||||
VMReg prev_vm_reg = VMRegImpl::Bad();
|
||||
|
||||
RegMaskIterator rmi(stub->live());
|
||||
while (rmi.has_next()) {
|
||||
const OptoReg::Name opto_reg = rmi.next();
|
||||
if (OptoReg::is_reg(opto_reg)) {
|
||||
const VMReg vm_reg = OptoReg::as_VMReg(opto_reg);
|
||||
OptoReg::Name opto_reg = rmi.next();
|
||||
VMReg vm_reg = OptoReg::as_VMReg(opto_reg);
|
||||
|
||||
if (vm_reg->is_Register()) {
|
||||
// GPR may have one or two slots in regmask
|
||||
// Determine whether the current vm_reg is the same physical register as the previous one
|
||||
if (is_same_register(vm_reg, prev_vm_reg)) {
|
||||
registers.at(index)._slots++;
|
||||
} else {
|
||||
RegisterData reg_data = { vm_reg, 1 };
|
||||
index = registers.append(reg_data);
|
||||
}
|
||||
} else if (vm_reg->is_FloatRegister()) {
|
||||
// We have size encoding in OptoReg of stub->live()
|
||||
// After encoding, float/neon/sve register has only one slot in regmask
|
||||
// Decode it to get the actual size
|
||||
VMReg vm_reg_base = vm_reg->as_FloatRegister()->as_VMReg();
|
||||
int slots = decode_float_vector_register_size(opto_reg);
|
||||
RegisterData reg_data = { vm_reg_base, slots };
|
||||
index = registers.append(reg_data);
|
||||
} else if (vm_reg->is_PRegister()) {
|
||||
// PRegister has only one slot in regmask
|
||||
RegisterData reg_data = { vm_reg, 1 };
|
||||
index = registers.append(reg_data);
|
||||
} else {
|
||||
assert(false, "Unknown register type");
|
||||
ShouldNotReachHere();
|
||||
}
|
||||
prev_vm_reg = vm_reg;
|
||||
}
|
||||
|
||||
// Record registers that needs to be saved/restored
|
||||
for (GrowableArrayIterator<RegisterData> it = registers.begin(); it != registers.end(); ++it) {
|
||||
RegisterData reg_data = *it;
|
||||
VMReg vm_reg = reg_data._reg;
|
||||
int slots = reg_data._slots;
|
||||
if (vm_reg->is_Register()) {
|
||||
assert(slots == 1 || slots == 2, "Unexpected register save size");
|
||||
_gp_regs += RegSet::of(vm_reg->as_Register());
|
||||
} else if (vm_reg->is_FloatRegister()) {
|
||||
if (slots == 1 || slots == 2) {
|
||||
_fp_regs += FloatRegSet::of(vm_reg->as_FloatRegister());
|
||||
} else if (vm_reg->is_PRegister()) {
|
||||
_p_regs += PRegSet::of(vm_reg->as_PRegister());
|
||||
} else if (slots == 4) {
|
||||
_neon_regs += FloatRegSet::of(vm_reg->as_FloatRegister());
|
||||
} else {
|
||||
fatal("Unknown register type");
|
||||
assert(slots == Matcher::scalable_vector_reg_size(T_FLOAT), "Unexpected register save size");
|
||||
_sve_regs += FloatRegSet::of(vm_reg->as_FloatRegister());
|
||||
}
|
||||
} else {
|
||||
assert(vm_reg->is_PRegister() && slots == 1, "Unknown register type");
|
||||
_p_regs += PRegSet::of(vm_reg->as_PRegister());
|
||||
}
|
||||
}
|
||||
|
||||
@ -1130,12 +1213,65 @@ public:
|
||||
} else {
|
||||
_gp_regs -= RegSet::range(r19, r30) + RegSet::of(r8, r9);
|
||||
}
|
||||
|
||||
// Remove C-ABI SOE fp registers
|
||||
_fp_regs -= FloatRegSet::range(v8, v15);
|
||||
}
|
||||
|
||||
static enum RC rc_class(VMReg reg) {
|
||||
if (reg->is_reg()) {
|
||||
if (reg->is_Register()) {
|
||||
return rc_int;
|
||||
} else if (reg->is_FloatRegister()) {
|
||||
return rc_float;
|
||||
} else if (reg->is_PRegister()) {
|
||||
return rc_predicate;
|
||||
}
|
||||
}
|
||||
if (reg->is_stack()) {
|
||||
return rc_stack;
|
||||
}
|
||||
return rc_bad;
|
||||
}
|
||||
|
||||
static bool is_same_register(VMReg reg1, VMReg reg2) {
|
||||
if (reg1 == reg2) {
|
||||
return true;
|
||||
}
|
||||
if (rc_class(reg1) == rc_class(reg2)) {
|
||||
if (reg1->is_Register()) {
|
||||
return reg1->as_Register() == reg2->as_Register();
|
||||
} else if (reg1->is_FloatRegister()) {
|
||||
return reg1->as_FloatRegister() == reg2->as_FloatRegister();
|
||||
} else if (reg1->is_PRegister()) {
|
||||
return reg1->as_PRegister() == reg2->as_PRegister();
|
||||
}
|
||||
}
|
||||
return false;
|
||||
}
|
||||
|
||||
static int decode_float_vector_register_size(OptoReg::Name opto_reg) {
|
||||
switch (opto_reg & 3) {
|
||||
case 0:
|
||||
return 1;
|
||||
case 1:
|
||||
return 2;
|
||||
case 2:
|
||||
return 4;
|
||||
case 3:
|
||||
return Matcher::scalable_vector_reg_size(T_FLOAT);
|
||||
default:
|
||||
ShouldNotReachHere();
|
||||
return 0;
|
||||
}
|
||||
}
|
||||
|
||||
ZSaveLiveRegisters(MacroAssembler* masm, ZBarrierStubC2* stub)
|
||||
: _masm(masm),
|
||||
_gp_regs(),
|
||||
_fp_regs(),
|
||||
_neon_regs(),
|
||||
_sve_regs(),
|
||||
_p_regs() {
|
||||
|
||||
// Figure out what registers to save/restore
|
||||
@ -1143,14 +1279,18 @@ public:
|
||||
|
||||
// Save registers
|
||||
__ push(_gp_regs, sp);
|
||||
__ push_fp(_fp_regs, sp);
|
||||
__ push_fp(_fp_regs, sp, MacroAssembler::PushPopFp);
|
||||
__ push_fp(_neon_regs, sp, MacroAssembler::PushPopNeon);
|
||||
__ push_fp(_sve_regs, sp, MacroAssembler::PushPopSVE);
|
||||
__ push_p(_p_regs, sp);
|
||||
}
|
||||
|
||||
~ZSaveLiveRegisters() {
|
||||
// Restore registers
|
||||
__ pop_p(_p_regs, sp);
|
||||
__ pop_fp(_fp_regs, sp);
|
||||
__ pop_fp(_sve_regs, sp, MacroAssembler::PushPopSVE);
|
||||
__ pop_fp(_neon_regs, sp, MacroAssembler::PushPopNeon);
|
||||
__ pop_fp(_fp_regs, sp, MacroAssembler::PushPopFp);
|
||||
|
||||
// External runtime call may clobber ptrue reg
|
||||
__ reinitialize_ptrue();
|
||||
|
@ -1,5 +1,5 @@
|
||||
/*
|
||||
* Copyright (c) 2019, 2023, Oracle and/or its affiliates. All rights reserved.
|
||||
* Copyright (c) 2019, 2024, Oracle and/or its affiliates. All rights reserved.
|
||||
* DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER.
|
||||
*
|
||||
* This code is free software; you can redistribute it and/or modify it
|
||||
@ -187,6 +187,9 @@ public:
|
||||
#endif // COMPILER1
|
||||
|
||||
#ifdef COMPILER2
|
||||
OptoReg::Name encode_float_vector_register_size(const Node* node,
|
||||
OptoReg::Name opto_reg);
|
||||
|
||||
OptoReg::Name refine_register(const Node* node,
|
||||
OptoReg::Name opto_reg);
|
||||
|
||||
|
@ -2641,7 +2641,7 @@ int MacroAssembler::pop(unsigned int bitset, Register stack) {
|
||||
|
||||
// Push lots of registers in the bit set supplied. Don't push sp.
|
||||
// Return the number of dwords pushed
|
||||
int MacroAssembler::push_fp(unsigned int bitset, Register stack) {
|
||||
int MacroAssembler::push_fp(unsigned int bitset, Register stack, FpPushPopMode mode) {
|
||||
int words_pushed = 0;
|
||||
bool use_sve = false;
|
||||
int sve_vector_size_in_bytes = 0;
|
||||
@ -2664,8 +2664,29 @@ int MacroAssembler::push_fp(unsigned int bitset, Register stack) {
|
||||
return 0;
|
||||
}
|
||||
|
||||
// SVE
|
||||
if (mode == PushPopFull) {
|
||||
if (use_sve && sve_vector_size_in_bytes > 16) {
|
||||
mode = PushPopSVE;
|
||||
} else {
|
||||
mode = PushPopNeon;
|
||||
}
|
||||
}
|
||||
|
||||
#ifndef PRODUCT
|
||||
{
|
||||
char buffer[48];
|
||||
if (mode == PushPopSVE) {
|
||||
snprintf(buffer, sizeof(buffer), "push_fp: %d SVE registers", count);
|
||||
} else if (mode == PushPopNeon) {
|
||||
snprintf(buffer, sizeof(buffer), "push_fp: %d Neon registers", count);
|
||||
} else {
|
||||
snprintf(buffer, sizeof(buffer), "push_fp: %d fp registers", count);
|
||||
}
|
||||
block_comment(buffer);
|
||||
}
|
||||
#endif
|
||||
|
||||
if (mode == PushPopSVE) {
|
||||
sub(stack, stack, sve_vector_size_in_bytes * count);
|
||||
for (int i = 0; i < count; i++) {
|
||||
sve_str(as_FloatRegister(regs[i]), Address(stack, i));
|
||||
@ -2673,7 +2694,7 @@ int MacroAssembler::push_fp(unsigned int bitset, Register stack) {
|
||||
return count * sve_vector_size_in_bytes / 8;
|
||||
}
|
||||
|
||||
// NEON
|
||||
if (mode == PushPopNeon) {
|
||||
if (count == 1) {
|
||||
strq(as_FloatRegister(regs[0]), Address(pre(stack, -wordSize * 2)));
|
||||
return 2;
|
||||
@ -2698,10 +2719,42 @@ int MacroAssembler::push_fp(unsigned int bitset, Register stack) {
|
||||
|
||||
assert(words_pushed == count, "oops, pushed(%d) != count(%d)", words_pushed, count);
|
||||
return count * 2;
|
||||
}
|
||||
|
||||
if (mode == PushPopFp) {
|
||||
bool odd = (count & 1) == 1;
|
||||
int push_slots = count + (odd ? 1 : 0);
|
||||
|
||||
if (count == 1) {
|
||||
// Stack pointer must be 16 bytes aligned
|
||||
strd(as_FloatRegister(regs[0]), Address(pre(stack, -push_slots * wordSize)));
|
||||
return 1;
|
||||
}
|
||||
|
||||
stpd(as_FloatRegister(regs[0]), as_FloatRegister(regs[1]), Address(pre(stack, -push_slots * wordSize)));
|
||||
words_pushed += 2;
|
||||
|
||||
for (int i = 2; i + 1 < count; i += 2) {
|
||||
stpd(as_FloatRegister(regs[i]), as_FloatRegister(regs[i+1]), Address(stack, i * wordSize));
|
||||
words_pushed += 2;
|
||||
}
|
||||
|
||||
if (odd) {
|
||||
// Stack pointer must be 16 bytes aligned
|
||||
strd(as_FloatRegister(regs[count - 1]), Address(stack, (count - 1) * wordSize));
|
||||
words_pushed++;
|
||||
}
|
||||
|
||||
assert(words_pushed == count, "oops, pushed != count");
|
||||
|
||||
return count;
|
||||
}
|
||||
|
||||
return 0;
|
||||
}
|
||||
|
||||
// Return the number of dwords popped
|
||||
int MacroAssembler::pop_fp(unsigned int bitset, Register stack) {
|
||||
int MacroAssembler::pop_fp(unsigned int bitset, Register stack, FpPushPopMode mode) {
|
||||
int words_pushed = 0;
|
||||
bool use_sve = false;
|
||||
int sve_vector_size_in_bytes = 0;
|
||||
@ -2723,8 +2776,29 @@ int MacroAssembler::pop_fp(unsigned int bitset, Register stack) {
|
||||
return 0;
|
||||
}
|
||||
|
||||
// SVE
|
||||
if (mode == PushPopFull) {
|
||||
if (use_sve && sve_vector_size_in_bytes > 16) {
|
||||
mode = PushPopSVE;
|
||||
} else {
|
||||
mode = PushPopNeon;
|
||||
}
|
||||
}
|
||||
|
||||
#ifndef PRODUCT
|
||||
{
|
||||
char buffer[48];
|
||||
if (mode == PushPopSVE) {
|
||||
snprintf(buffer, sizeof(buffer), "pop_fp: %d SVE registers", count);
|
||||
} else if (mode == PushPopNeon) {
|
||||
snprintf(buffer, sizeof(buffer), "pop_fp: %d Neon registers", count);
|
||||
} else {
|
||||
snprintf(buffer, sizeof(buffer), "pop_fp: %d fp registers", count);
|
||||
}
|
||||
block_comment(buffer);
|
||||
}
|
||||
#endif
|
||||
|
||||
if (mode == PushPopSVE) {
|
||||
for (int i = count - 1; i >= 0; i--) {
|
||||
sve_ldr(as_FloatRegister(regs[i]), Address(stack, i));
|
||||
}
|
||||
@ -2732,7 +2806,7 @@ int MacroAssembler::pop_fp(unsigned int bitset, Register stack) {
|
||||
return count * sve_vector_size_in_bytes / 8;
|
||||
}
|
||||
|
||||
// NEON
|
||||
if (mode == PushPopNeon) {
|
||||
if (count == 1) {
|
||||
ldrq(as_FloatRegister(regs[0]), Address(post(stack, wordSize * 2)));
|
||||
return 2;
|
||||
@ -2757,6 +2831,36 @@ int MacroAssembler::pop_fp(unsigned int bitset, Register stack) {
|
||||
assert(words_pushed == count, "oops, pushed(%d) != count(%d)", words_pushed, count);
|
||||
|
||||
return count * 2;
|
||||
}
|
||||
|
||||
if (mode == PushPopFp) {
|
||||
bool odd = (count & 1) == 1;
|
||||
int push_slots = count + (odd ? 1 : 0);
|
||||
|
||||
if (count == 1) {
|
||||
ldrd(as_FloatRegister(regs[0]), Address(post(stack, push_slots * wordSize)));
|
||||
return 1;
|
||||
}
|
||||
|
||||
if (odd) {
|
||||
ldrd(as_FloatRegister(regs[count - 1]), Address(stack, (count - 1) * wordSize));
|
||||
words_pushed++;
|
||||
}
|
||||
|
||||
for (int i = 2; i + 1 < count; i += 2) {
|
||||
ldpd(as_FloatRegister(regs[i]), as_FloatRegister(regs[i+1]), Address(stack, i * wordSize));
|
||||
words_pushed += 2;
|
||||
}
|
||||
|
||||
ldpd(as_FloatRegister(regs[0]), as_FloatRegister(regs[1]), Address(post(stack, push_slots * wordSize)));
|
||||
words_pushed += 2;
|
||||
|
||||
assert(words_pushed == count, "oops, pushed != count");
|
||||
|
||||
return count;
|
||||
}
|
||||
|
||||
return 0;
|
||||
}
|
||||
|
||||
// Return the number of dwords pushed
|
||||
|
@ -444,6 +444,15 @@ class MacroAssembler: public Assembler {
|
||||
|
||||
// macro assembly operations needed for aarch64
|
||||
|
||||
public:
|
||||
|
||||
enum FpPushPopMode {
|
||||
PushPopFull,
|
||||
PushPopSVE,
|
||||
PushPopNeon,
|
||||
PushPopFp
|
||||
};
|
||||
|
||||
// first two private routines for loading 32 bit or 64 bit constants
|
||||
private:
|
||||
|
||||
@ -453,8 +462,8 @@ private:
|
||||
int push(unsigned int bitset, Register stack);
|
||||
int pop(unsigned int bitset, Register stack);
|
||||
|
||||
int push_fp(unsigned int bitset, Register stack);
|
||||
int pop_fp(unsigned int bitset, Register stack);
|
||||
int push_fp(unsigned int bitset, Register stack, FpPushPopMode mode);
|
||||
int pop_fp(unsigned int bitset, Register stack, FpPushPopMode mode);
|
||||
|
||||
int push_p(unsigned int bitset, Register stack);
|
||||
int pop_p(unsigned int bitset, Register stack);
|
||||
@ -462,11 +471,12 @@ private:
|
||||
void mov(Register dst, Address a);
|
||||
|
||||
public:
|
||||
|
||||
void push(RegSet regs, Register stack) { if (regs.bits()) push(regs.bits(), stack); }
|
||||
void pop(RegSet regs, Register stack) { if (regs.bits()) pop(regs.bits(), stack); }
|
||||
|
||||
void push_fp(FloatRegSet regs, Register stack) { if (regs.bits()) push_fp(regs.bits(), stack); }
|
||||
void pop_fp(FloatRegSet regs, Register stack) { if (regs.bits()) pop_fp(regs.bits(), stack); }
|
||||
void push_fp(FloatRegSet regs, Register stack, FpPushPopMode mode = PushPopFull) { if (regs.bits()) push_fp(regs.bits(), stack, mode); }
|
||||
void pop_fp(FloatRegSet regs, Register stack, FpPushPopMode mode = PushPopFull) { if (regs.bits()) pop_fp(regs.bits(), stack, mode); }
|
||||
|
||||
static RegSet call_clobbered_gp_registers();
|
||||
|
||||
|
@ -1,5 +1,5 @@
|
||||
/*
|
||||
* Copyright (c) 2006, 2021, Oracle and/or its affiliates. All rights reserved.
|
||||
* Copyright (c) 2006, 2024, Oracle and/or its affiliates. All rights reserved.
|
||||
* Copyright (c) 2014, Red Hat Inc. All rights reserved.
|
||||
* DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER.
|
||||
*
|
||||
@ -45,7 +45,7 @@ inline Register as_Register() {
|
||||
}
|
||||
|
||||
inline FloatRegister as_FloatRegister() {
|
||||
assert( is_FloatRegister() && is_even(value()), "must be" );
|
||||
assert( is_FloatRegister(), "must be" );
|
||||
// Yuk
|
||||
return ::as_FloatRegister((value() - ConcreteRegisterImpl::max_gpr) /
|
||||
FloatRegister::max_slots_per_register);
|
||||
|
@ -0,0 +1,380 @@
|
||||
/*
|
||||
* Copyright (c) 2024, Alibaba Group Holding Limited. All Rights Reserved.
|
||||
* DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER.
|
||||
*
|
||||
* This code is free software; you can redistribute it and/or modify it
|
||||
* under the terms of the GNU General Public License version 2 only, as
|
||||
* published by the Free Software Foundation.
|
||||
*
|
||||
* This code is distributed in the hope that it will be useful, but WITHOUT
|
||||
* ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or
|
||||
* FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License
|
||||
* version 2 for more details (a copy is included in the LICENSE file that
|
||||
* accompanied this code).
|
||||
*
|
||||
* You should have received a copy of the GNU General Public License version
|
||||
* 2 along with this work; if not, write to the Free Software Foundation,
|
||||
* Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA.
|
||||
*
|
||||
* Please contact Oracle, 500 Oracle Parkway, Redwood Shores, CA 94065 USA
|
||||
* or visit www.oracle.com if you need additional information or have any
|
||||
* questions.
|
||||
*/
|
||||
|
||||
package gc.z;
|
||||
|
||||
/**
|
||||
* @test TestRegistersPushPopAtZGCLoadBarrierStub
|
||||
* @bug 8326541
|
||||
* @summary Test to verify that registers are saved and restored correctly based on
|
||||
the actual register usage length on aarch64 when entering load barrier stub.
|
||||
* @library /test/lib /
|
||||
* @modules jdk.incubator.vector
|
||||
*
|
||||
* @requires vm.gc.ZGenerational & vm.debug
|
||||
* @requires os.arch=="aarch64"
|
||||
*
|
||||
* @run driver gc.z.TestRegistersPushPopAtZGCLoadBarrierStub
|
||||
*/
|
||||
|
||||
import java.util.ArrayList;
|
||||
import java.util.LinkedHashMap;
|
||||
import java.util.Map;
|
||||
import java.util.random.RandomGenerator;
|
||||
import java.util.random.RandomGeneratorFactory;
|
||||
import java.util.regex.Matcher;
|
||||
import java.util.regex.Pattern;
|
||||
import jdk.incubator.vector.FloatVector;
|
||||
import jdk.incubator.vector.Vector;
|
||||
import jdk.incubator.vector.VectorShape;
|
||||
import jdk.incubator.vector.VectorSpecies;
|
||||
|
||||
import jdk.test.lib.Asserts;
|
||||
import jdk.test.lib.process.OutputAnalyzer;
|
||||
import jdk.test.lib.process.ProcessTools;
|
||||
|
||||
class Inner {}
|
||||
|
||||
class InnerFloat extends Inner {
|
||||
float data;
|
||||
public InnerFloat(float f) {
|
||||
data = f;
|
||||
}
|
||||
}
|
||||
|
||||
class InnerDouble extends Inner {
|
||||
double data;
|
||||
public InnerDouble(double f) {
|
||||
data = f;
|
||||
}
|
||||
}
|
||||
|
||||
class Outer {
|
||||
volatile Inner field;
|
||||
public Outer(Inner i) {
|
||||
field = i;
|
||||
}
|
||||
}
|
||||
|
||||
public class TestRegistersPushPopAtZGCLoadBarrierStub {
|
||||
|
||||
class Launcher {
|
||||
private final static int NUM = 1024;
|
||||
private final static int ITERATIONS = 20_000;
|
||||
private final static RandomGenerator RANDOM = RandomGeneratorFactory.getDefault().create(0);
|
||||
private final static Map<String, Runnable> TESTS;
|
||||
|
||||
private static float[] f_array;
|
||||
private static Outer f_outer;
|
||||
private static Outer d_outer;
|
||||
|
||||
static {
|
||||
f_array = new float[NUM];
|
||||
for (int i = 0; i < NUM; i++) {
|
||||
f_array[i] = RANDOM.nextFloat();
|
||||
}
|
||||
|
||||
InnerFloat f_inner = new InnerFloat(RANDOM.nextFloat());
|
||||
InnerDouble d_inner = new InnerDouble(RANDOM.nextDouble());
|
||||
f_outer = new Outer(f_inner);
|
||||
d_outer = new Outer(d_inner);
|
||||
|
||||
TESTS = new LinkedHashMap<>();
|
||||
TESTS.put("test_one_float_push_pop_at_load_barrier", Launcher::test_one_float);
|
||||
TESTS.put("test_two_floats_push_pop_at_load_barrier", Launcher::test_two_floats);
|
||||
TESTS.put("test_three_floats_push_pop_at_load_barrier", Launcher::test_three_floats);
|
||||
TESTS.put("test_one_double_push_pop_at_load_barrier", Launcher::test_one_double);
|
||||
TESTS.put("test_two_doubles_push_pop_at_load_barrier", Launcher::test_two_doubles);
|
||||
TESTS.put("test_three_doubles_push_pop_at_load_barrier", Launcher::test_three_doubles);
|
||||
TESTS.put("test_one_vector_128_push_pop_at_load_barrier", Launcher::test_one_vector_128);
|
||||
TESTS.put("test_two_vectors_128_push_pop_at_load_barrier", Launcher::test_two_vectors_128);
|
||||
TESTS.put("test_three_vectors_128_push_pop_at_load_barrier", Launcher::test_three_vectors_128);
|
||||
TESTS.put("test_vector_max_push_pop_at_load_barrier", Launcher::test_vector_max);
|
||||
TESTS.put("test_float_and_vector_push_pop_at_load_barrier", Launcher::test_float_and_vector);
|
||||
}
|
||||
|
||||
static float test_one_float_push_pop_at_load_barrier(Outer outer, float f) {
|
||||
Inner inner = outer.field;
|
||||
return f + ((InnerFloat)inner).data;
|
||||
}
|
||||
|
||||
static float test_two_floats_push_pop_at_load_barrier(Outer outer, float f1, float f2) {
|
||||
Inner inner = outer.field;
|
||||
return f1 + f2 + ((InnerFloat)inner).data;
|
||||
}
|
||||
|
||||
static float test_three_floats_push_pop_at_load_barrier(Outer outer, float f1, float f2, float f3) {
|
||||
Inner inner = outer.field;
|
||||
return f1 + f2 + f3 + ((InnerFloat)inner).data;
|
||||
}
|
||||
|
||||
static double test_one_double_push_pop_at_load_barrier(Outer outer, double d) {
|
||||
Inner inner = outer.field;
|
||||
return d + ((InnerDouble)inner).data;
|
||||
}
|
||||
|
||||
static double test_two_doubles_push_pop_at_load_barrier(Outer outer, double d1, double d2) {
|
||||
Inner inner = outer.field;
|
||||
return d1 + d2 + ((InnerDouble)inner).data;
|
||||
}
|
||||
|
||||
static double test_three_doubles_push_pop_at_load_barrier(Outer outer, double d1, double d2, double d3) {
|
||||
Inner inner = outer.field;
|
||||
return d1 + d2 + d3 + ((InnerDouble)inner).data;
|
||||
}
|
||||
|
||||
static void test_one_vector_128_push_pop_at_load_barrier(float[] b, Outer outer) {
|
||||
VectorSpecies<Float> float_species = FloatVector.SPECIES_128;
|
||||
|
||||
FloatVector av = FloatVector.zero(float_species);
|
||||
for (int i = 0; i < b.length; i += float_species.length()) {
|
||||
Inner inner = outer.field;
|
||||
FloatVector bv = FloatVector.fromArray(float_species, b, i);
|
||||
float value = ((InnerFloat)inner).data;
|
||||
av = av.add(bv).add(value);
|
||||
}
|
||||
}
|
||||
|
||||
static void test_two_vectors_128_push_pop_at_load_barrier(float[] b, Outer outer) {
|
||||
VectorSpecies<Float> float_species = FloatVector.SPECIES_128;
|
||||
|
||||
FloatVector av1 = FloatVector.zero(float_species);
|
||||
FloatVector av2 = FloatVector.zero(float_species);
|
||||
for (int i = 0; i < b.length; i += float_species.length()) {
|
||||
Inner inner = outer.field;
|
||||
FloatVector bv = FloatVector.fromArray(float_species, b, i);
|
||||
float value = ((InnerFloat)inner).data;
|
||||
av1 = av1.add(bv).add(value);
|
||||
av2 = av2.add(av1);
|
||||
}
|
||||
}
|
||||
|
||||
static void test_three_vectors_128_push_pop_at_load_barrier(float[] b, Outer outer) {
|
||||
VectorSpecies<Float> float_species = FloatVector.SPECIES_128;
|
||||
|
||||
FloatVector av1 = FloatVector.zero(float_species);
|
||||
FloatVector av2 = FloatVector.zero(float_species);
|
||||
FloatVector av3 = FloatVector.zero(float_species);
|
||||
for (int i = 0; i < b.length; i += float_species.length()) {
|
||||
Inner inner = outer.field;
|
||||
FloatVector bv = FloatVector.fromArray(float_species, b, i);
|
||||
float value = ((InnerFloat)inner).data;
|
||||
av1 = av1.add(bv).add(value);
|
||||
av2 = av2.add(av1);
|
||||
av3 = av3.add(av2);
|
||||
}
|
||||
}
|
||||
|
||||
static void test_vector_max_push_pop_at_load_barrier(float[] b, Outer outer) {
|
||||
VectorSpecies<Float> float_species = FloatVector.SPECIES_MAX;
|
||||
|
||||
FloatVector av = FloatVector.zero(float_species);
|
||||
for (int i = 0; i < b.length; i += float_species.length()) {
|
||||
Inner inner = outer.field;
|
||||
FloatVector bv = FloatVector.fromArray(float_species, b, i);
|
||||
float value = ((InnerFloat)inner).data;
|
||||
av = av.add(bv).add(value);
|
||||
}
|
||||
}
|
||||
|
||||
static void test_float_and_vector_push_pop_at_load_barrier(float[] b, Outer outer, float f) {
|
||||
VectorSpecies<Float> float_species = FloatVector.SPECIES_MAX;
|
||||
|
||||
FloatVector av = FloatVector.zero(float_species);
|
||||
for (int i = 0; i < b.length; i += float_species.length()) {
|
||||
Inner inner = outer.field;
|
||||
FloatVector bv = FloatVector.fromArray(float_species, b, i);
|
||||
float value = ((InnerFloat)inner).data + f;
|
||||
av = av.add(bv).add(value);
|
||||
}
|
||||
}
|
||||
|
||||
static void test_one_float() {
|
||||
for (int i = 0; i < ITERATIONS; i++) {
|
||||
test_one_float_push_pop_at_load_barrier(f_outer, RANDOM.nextFloat());
|
||||
}
|
||||
}
|
||||
|
||||
static void test_two_floats() {
|
||||
for (int i = 0; i < ITERATIONS; i++) {
|
||||
test_two_floats_push_pop_at_load_barrier(f_outer, RANDOM.nextFloat(), RANDOM.nextFloat());
|
||||
}
|
||||
}
|
||||
|
||||
static void test_three_floats() {
|
||||
for (int i = 0; i < ITERATIONS; i++) {
|
||||
test_three_floats_push_pop_at_load_barrier(f_outer, RANDOM.nextFloat(), RANDOM.nextFloat(), RANDOM.nextFloat());
|
||||
}
|
||||
}
|
||||
|
||||
static void test_one_double() {
|
||||
for (int i = 0; i < ITERATIONS; i++) {
|
||||
test_one_double_push_pop_at_load_barrier(d_outer, RANDOM.nextDouble());
|
||||
}
|
||||
}
|
||||
|
||||
static void test_two_doubles() {
|
||||
for (int i = 0; i < ITERATIONS; i++) {
|
||||
test_two_doubles_push_pop_at_load_barrier(d_outer, RANDOM.nextDouble(), RANDOM.nextDouble());
|
||||
}
|
||||
}
|
||||
|
||||
static void test_three_doubles() {
|
||||
for (int i = 0; i < ITERATIONS; i++) {
|
||||
test_three_doubles_push_pop_at_load_barrier(d_outer, RANDOM.nextDouble(), RANDOM.nextDouble(), RANDOM.nextDouble());
|
||||
}
|
||||
}
|
||||
|
||||
static void test_one_vector_128() {
|
||||
for (int i = 0; i < ITERATIONS; i++) {
|
||||
test_one_vector_128_push_pop_at_load_barrier(f_array, f_outer);
|
||||
}
|
||||
}
|
||||
|
||||
static void test_two_vectors_128() {
|
||||
for (int i = 0; i < ITERATIONS; i++) {
|
||||
test_two_vectors_128_push_pop_at_load_barrier(f_array, f_outer);
|
||||
}
|
||||
}
|
||||
|
||||
static void test_three_vectors_128() {
|
||||
for (int i = 0; i < ITERATIONS; i++) {
|
||||
test_three_vectors_128_push_pop_at_load_barrier(f_array, f_outer);
|
||||
}
|
||||
}
|
||||
|
||||
static void test_vector_max() {
|
||||
for (int i = 0; i < ITERATIONS; i++) {
|
||||
test_vector_max_push_pop_at_load_barrier(f_array, f_outer);
|
||||
}
|
||||
}
|
||||
|
||||
static void test_float_and_vector() {
|
||||
for (int i = 0; i < ITERATIONS; i++) {
|
||||
test_float_and_vector_push_pop_at_load_barrier(f_array, f_outer, RANDOM.nextFloat());
|
||||
}
|
||||
}
|
||||
|
||||
public static void main(String args[]) {
|
||||
Runnable r = TESTS.get(args[0]);
|
||||
r.run();
|
||||
}
|
||||
}
|
||||
|
||||
static boolean containOnlyOneOccuranceOfKeyword(String text, String keyword) {
|
||||
int firstIndex = text.indexOf(keyword);
|
||||
int lastIndex = text.lastIndexOf(keyword);
|
||||
return firstIndex != -1 && firstIndex == lastIndex;
|
||||
}
|
||||
|
||||
// Check that registers are pushed and poped with correct register type and number
|
||||
static void checkPushPopRegNumberAndType(String stdout, String keyword, String expected_freg_type,
|
||||
int expected_number_of_fregs) throws Exception {
|
||||
String expected = keyword + expected_number_of_fregs + " " + expected_freg_type + " registers";
|
||||
|
||||
String regex = keyword + "(\\d+) " + expected_freg_type + " registers";
|
||||
Pattern p = Pattern.compile(regex);
|
||||
Matcher m = p.matcher(stdout);
|
||||
|
||||
if (m.find()) {
|
||||
String found = m.group();
|
||||
Asserts.assertEquals(found, expected, "found '" + found + "' but should print '" + expected + "'");
|
||||
} else {
|
||||
throw new RuntimeException("'" + regex + "' is not found in stdout");
|
||||
}
|
||||
|
||||
if (m.find()) {
|
||||
throw new RuntimeException("Stdout is expected to contain only one occurance of '" + regex +
|
||||
"'. Found another occurance: '" + m.group() + "'");
|
||||
}
|
||||
}
|
||||
|
||||
static String launchJavaTestProcess(String test_name) throws Exception {
|
||||
ArrayList<String> command = new ArrayList<String>();
|
||||
command.add("-Xbatch");
|
||||
command.add("-XX:LoopUnrollLimit=0");
|
||||
command.add("-XX:-UseOnStackReplacement");
|
||||
command.add("-XX:-TieredCompilation");
|
||||
command.add("-XX:+UseZGC");
|
||||
command.add("-XX:+ZGenerational");
|
||||
command.add("--add-modules=jdk.incubator.vector");
|
||||
command.add("-XX:CompileCommand=print," + Launcher.class.getName() + "::" + test_name);
|
||||
command.add(Launcher.class.getName());
|
||||
command.add(test_name);
|
||||
|
||||
ProcessBuilder pb = ProcessTools.createLimitedTestJavaProcessBuilder(command);
|
||||
OutputAnalyzer output = new OutputAnalyzer(pb.start());
|
||||
output.shouldHaveExitValue(0);
|
||||
return output.getStdout();
|
||||
}
|
||||
|
||||
static void run_test(String test_name, String expected_freg_type, int expected_number_of_fregs,
|
||||
String expected_vector_reg_type, int expected_number_of_vector_regs) throws Exception {
|
||||
String stdout = launchJavaTestProcess(test_name);
|
||||
|
||||
String keyword = "push_fp: ";
|
||||
checkPushPopRegNumberAndType(stdout, keyword, expected_freg_type, expected_number_of_fregs);
|
||||
checkPushPopRegNumberAndType(stdout, keyword, expected_vector_reg_type, expected_number_of_vector_regs);
|
||||
|
||||
keyword = "pop_fp: ";
|
||||
checkPushPopRegNumberAndType(stdout, keyword, expected_freg_type, expected_number_of_fregs);
|
||||
checkPushPopRegNumberAndType(stdout, keyword, expected_vector_reg_type, expected_number_of_vector_regs);
|
||||
}
|
||||
|
||||
static void run_test(String test_name, String expected_freg_type, int expected_number_of_fregs) throws Exception {
|
||||
String stdout = launchJavaTestProcess(test_name);
|
||||
|
||||
String keyword = "push_fp: ";
|
||||
if (!containOnlyOneOccuranceOfKeyword(stdout, keyword)) {
|
||||
throw new RuntimeException("Stdout is expected to contain only one occurance of keyword: " + "'" + keyword + "'");
|
||||
}
|
||||
checkPushPopRegNumberAndType(stdout, keyword, expected_freg_type, expected_number_of_fregs);
|
||||
|
||||
keyword = "pop_fp: ";
|
||||
if (!containOnlyOneOccuranceOfKeyword(stdout, keyword)) {
|
||||
throw new RuntimeException("Stdout is expected to contain only one occurance of keyword: " + "'" + keyword + "'");
|
||||
}
|
||||
checkPushPopRegNumberAndType(stdout, keyword, expected_freg_type, expected_number_of_fregs);
|
||||
}
|
||||
|
||||
public static void main(String[] args) throws Exception {
|
||||
String vector_max_reg_type;
|
||||
if (VectorShape.S_Max_BIT.vectorBitSize() > 128) {
|
||||
vector_max_reg_type = "SVE";
|
||||
} else {
|
||||
vector_max_reg_type = "Neon";
|
||||
}
|
||||
run_test("test_one_float_push_pop_at_load_barrier", "fp", 1);
|
||||
run_test("test_two_floats_push_pop_at_load_barrier", "fp", 2);
|
||||
run_test("test_three_floats_push_pop_at_load_barrier", "fp", 3);
|
||||
run_test("test_one_double_push_pop_at_load_barrier", "fp", 1);
|
||||
run_test("test_two_doubles_push_pop_at_load_barrier", "fp", 2);
|
||||
run_test("test_three_doubles_push_pop_at_load_barrier", "fp", 3);
|
||||
run_test("test_one_vector_128_push_pop_at_load_barrier", "Neon", 1);
|
||||
run_test("test_two_vectors_128_push_pop_at_load_barrier", "Neon", 2);
|
||||
run_test("test_three_vectors_128_push_pop_at_load_barrier", "Neon", 3);
|
||||
run_test("test_vector_max_push_pop_at_load_barrier", vector_max_reg_type, 1);
|
||||
run_test("test_float_and_vector_push_pop_at_load_barrier", "fp", 1, vector_max_reg_type, 1);
|
||||
}
|
||||
}
|
||||
|
Loading…
Reference in New Issue
Block a user