8339741: RISC-V: C ABI breakage for integer on stack
Reviewed-by: fyang, luhenry
This commit is contained in:
parent
55a7cf1445
commit
bfe7f9205b
@ -74,6 +74,16 @@ InterpreterRuntime::SignatureHandlerGenerator::SignatureHandlerGenerator(
|
|||||||
_stack_offset = 0;
|
_stack_offset = 0;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// The C ABI specifies:
|
||||||
|
// "integer scalars narrower than XLEN bits are widened according to the sign
|
||||||
|
// of their type up to 32 bits, then sign-extended to XLEN bits."
|
||||||
|
// Applies for both passed in register and stack.
|
||||||
|
//
|
||||||
|
// Java uses 32-bit stack slots; jint, jshort, jchar, jbyte uses one slot.
|
||||||
|
// Native uses 64-bit stack slots for all integer scalar types.
|
||||||
|
//
|
||||||
|
// lw loads the Java stack slot, sign-extends and
|
||||||
|
// sd store this widened integer into a 64 bit native stack slot.
|
||||||
void InterpreterRuntime::SignatureHandlerGenerator::pass_int() {
|
void InterpreterRuntime::SignatureHandlerGenerator::pass_int() {
|
||||||
const Address src(from(), Interpreter::local_offset_in_bytes(offset()));
|
const Address src(from(), Interpreter::local_offset_in_bytes(offset()));
|
||||||
|
|
||||||
@ -82,7 +92,7 @@ void InterpreterRuntime::SignatureHandlerGenerator::pass_int() {
|
|||||||
__ lw(reg, src);
|
__ lw(reg, src);
|
||||||
} else {
|
} else {
|
||||||
__ lw(x10, src);
|
__ lw(x10, src);
|
||||||
__ sw(x10, Address(to(), next_stack_offset()));
|
__ sd(x10, Address(to(), next_stack_offset()));
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -5526,15 +5526,21 @@ static int reg2offset_out(VMReg r) {
|
|||||||
return (r->reg2stack() + SharedRuntime::out_preserve_stack_slots()) * VMRegImpl::stack_slot_size;
|
return (r->reg2stack() + SharedRuntime::out_preserve_stack_slots()) * VMRegImpl::stack_slot_size;
|
||||||
}
|
}
|
||||||
|
|
||||||
// On 64 bit we will store integer like items to the stack as
|
// The C ABI specifies:
|
||||||
// 64 bits items (riscv64 abi) even though java would only store
|
// "integer scalars narrower than XLEN bits are widened according to the sign
|
||||||
// 32bits for a parameter. On 32bit it will simply be 32 bits
|
// of their type up to 32 bits, then sign-extended to XLEN bits."
|
||||||
// So this routine will do 32->32 on 32bit and 32->64 on 64bit
|
// Applies for both passed in register and stack.
|
||||||
|
//
|
||||||
|
// Java uses 32-bit stack slots; jint, jshort, jchar, jbyte uses one slot.
|
||||||
|
// Native uses 64-bit stack slots for all integer scalar types.
|
||||||
|
//
|
||||||
|
// lw loads the Java stack slot, sign-extends and
|
||||||
|
// sd store this widened integer into a 64 bit native stack slot.
|
||||||
void MacroAssembler::move32_64(VMRegPair src, VMRegPair dst, Register tmp) {
|
void MacroAssembler::move32_64(VMRegPair src, VMRegPair dst, Register tmp) {
|
||||||
if (src.first()->is_stack()) {
|
if (src.first()->is_stack()) {
|
||||||
if (dst.first()->is_stack()) {
|
if (dst.first()->is_stack()) {
|
||||||
// stack to stack
|
// stack to stack
|
||||||
ld(tmp, Address(fp, reg2offset_in(src.first())));
|
lw(tmp, Address(fp, reg2offset_in(src.first())));
|
||||||
sd(tmp, Address(sp, reg2offset_out(dst.first())));
|
sd(tmp, Address(sp, reg2offset_out(dst.first())));
|
||||||
} else {
|
} else {
|
||||||
// stack to reg
|
// stack to reg
|
||||||
|
65
test/hotspot/jtreg/compiler/calls/TestManyArgs.java
Normal file
65
test/hotspot/jtreg/compiler/calls/TestManyArgs.java
Normal file
@ -0,0 +1,65 @@
|
|||||||
|
/*
|
||||||
|
* Copyright (c) 2024, Oracle and/or its affiliates. All rights reserved.
|
||||||
|
* Copyright (c) 2024, Rivos Inc. 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.
|
||||||
|
*/
|
||||||
|
|
||||||
|
/* @test
|
||||||
|
* @summary Pass values on stack.
|
||||||
|
* @requires os.arch == "riscv64"
|
||||||
|
* @run main/native compiler.calls.TestManyArgs
|
||||||
|
*/
|
||||||
|
|
||||||
|
package compiler.calls;
|
||||||
|
|
||||||
|
public class TestManyArgs {
|
||||||
|
static {
|
||||||
|
System.loadLibrary("TestManyArgs");
|
||||||
|
}
|
||||||
|
|
||||||
|
native static void scramblestack();
|
||||||
|
|
||||||
|
native static int checkargs(int arg0, short arg1, byte arg2,
|
||||||
|
int arg3, short arg4, byte arg5,
|
||||||
|
int arg6, short arg7, byte arg8,
|
||||||
|
int arg9, short arg10, byte arg11);
|
||||||
|
|
||||||
|
static int compiledbridge(int arg0, short arg1, byte arg2,
|
||||||
|
int arg3, short arg4, byte arg5,
|
||||||
|
int arg6, short arg7, byte arg8,
|
||||||
|
int arg9, short arg10, byte arg11) {
|
||||||
|
return checkargs(arg0, arg1, arg2, arg3, arg4, arg5,
|
||||||
|
arg6, arg7, arg8, arg9, arg10, arg11);
|
||||||
|
}
|
||||||
|
|
||||||
|
static public void main(String[] args) {
|
||||||
|
scramblestack();
|
||||||
|
for (int i = 0; i < 20000; i++) {
|
||||||
|
int res = compiledbridge((int)0xf, (short)0xf, (byte)0xf,
|
||||||
|
(int)0xf, (short)0xf, (byte)0xf,
|
||||||
|
(int)0xf, (short)0xf, (byte)0xf,
|
||||||
|
(int)0xf, (short)0xf, (byte)0xf);
|
||||||
|
if (res != 0) {
|
||||||
|
throw new RuntimeException("Test failed");
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
69
test/hotspot/jtreg/compiler/calls/libTestManyArgs.c
Normal file
69
test/hotspot/jtreg/compiler/calls/libTestManyArgs.c
Normal file
@ -0,0 +1,69 @@
|
|||||||
|
/*
|
||||||
|
* Copyright (c) 2024, Oracle and/or its affiliates. All rights reserved.
|
||||||
|
* Copyright (c) 2024, Rivos Inc. 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.
|
||||||
|
*/
|
||||||
|
|
||||||
|
#include "jni.h"
|
||||||
|
|
||||||
|
#ifdef riscv64
|
||||||
|
/* RV64 ABI pass all integers as 64-bit, in registers or on stack
|
||||||
|
* As compiler may choose to load smaller width than 64-bit if passed on stack,
|
||||||
|
* this test may not find any bugs.
|
||||||
|
* Therefore we trick the compiler todo 64-bit loads,
|
||||||
|
* by saying these args are jlongs.
|
||||||
|
*/
|
||||||
|
JNIEXPORT jint JNICALL Java_compiler_calls_TestManyArgs_checkargs(JNIEnv* env, jclass jclazz,
|
||||||
|
jlong arg0, jlong arg1, jlong arg2,
|
||||||
|
jlong arg3, jlong arg4, jlong arg5,
|
||||||
|
jlong arg6, jlong arg7, jlong arg8,
|
||||||
|
jlong arg9, jlong arg10, jlong arg11)
|
||||||
|
#else
|
||||||
|
JNIEXPORT jint JNICALL Java_compiler_calls_TestManyArgs_checkargs(JNIEnv* env, jclass jclazz,
|
||||||
|
jint arg0, jshort arg1, jbyte arg2,
|
||||||
|
jint arg3, jshort arg4, jbyte arg5,
|
||||||
|
jint arg6, jshort arg7, jbyte arg8,
|
||||||
|
jint arg9, jshort arg10, jbyte arg11)
|
||||||
|
#endif
|
||||||
|
{
|
||||||
|
if (arg0 != 0xf) return 1;
|
||||||
|
if (arg1 != 0xf) return 1;
|
||||||
|
if (arg2 != 0xf) return 1;
|
||||||
|
if (arg3 != 0xf) return 1;
|
||||||
|
if (arg4 != 0xf) return 1;
|
||||||
|
if (arg5 != 0xf) return 1;
|
||||||
|
if (arg6 != 0xf) return 1;
|
||||||
|
if (arg7 != 0xf) return 1;
|
||||||
|
if (arg8 != 0xf) return 1;
|
||||||
|
if (arg9 != 0xf) return 1;
|
||||||
|
if (arg10 != 0xf) return 1;
|
||||||
|
if (arg11 != 0xf) return 1;
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
JNIEXPORT
|
||||||
|
void JNICALL Java_compiler_calls_TestManyArgs_scramblestack(JNIEnv* env, jclass jclazz)
|
||||||
|
{
|
||||||
|
volatile char stack[12*8];
|
||||||
|
for (unsigned int i = 0; i < sizeof(stack); i++) {
|
||||||
|
stack[i] = (char)0xff;
|
||||||
|
}
|
||||||
|
}
|
Loading…
x
Reference in New Issue
Block a user