8339741: RISC-V: C ABI breakage for integer on stack

Reviewed-by: fyang, luhenry
This commit is contained in:
Robbin Ehn 2024-09-11 16:08:24 +00:00
parent 55a7cf1445
commit bfe7f9205b
4 changed files with 156 additions and 6 deletions

View File

@ -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()));
} }
} }

View File

@ -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

View 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");
}
}
}
}

View 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;
}
}