8255150: Add utility methods to check long indexes and ranges
Co-authored-by: Paul Sandoz <psandoz@openjdk.org> Reviewed-by: jvernee, dlong, vlivanov
This commit is contained in:
parent
6d878565f8
commit
a7422ac2f4
@ -8860,6 +8860,17 @@ instruct castII(iRegI dst)
|
|||||||
ins_pipe(pipe_class_empty);
|
ins_pipe(pipe_class_empty);
|
||||||
%}
|
%}
|
||||||
|
|
||||||
|
instruct castLL(iRegL dst)
|
||||||
|
%{
|
||||||
|
match(Set dst (CastLL dst));
|
||||||
|
|
||||||
|
size(0);
|
||||||
|
format %{ "# castLL of $dst" %}
|
||||||
|
ins_encode(/* empty encoding */);
|
||||||
|
ins_cost(0);
|
||||||
|
ins_pipe(pipe_class_empty);
|
||||||
|
%}
|
||||||
|
|
||||||
// ============================================================================
|
// ============================================================================
|
||||||
// Atomic operation instructions
|
// Atomic operation instructions
|
||||||
//
|
//
|
||||||
|
@ -5277,6 +5277,14 @@ instruct castII( iRegI dst ) %{
|
|||||||
ins_pipe(empty);
|
ins_pipe(empty);
|
||||||
%}
|
%}
|
||||||
|
|
||||||
|
instruct castLL( iRegL dst ) %{
|
||||||
|
match(Set dst (CastLL dst));
|
||||||
|
format %{ "! castLL of $dst" %}
|
||||||
|
ins_encode( /*empty encoding*/ );
|
||||||
|
ins_cost(0);
|
||||||
|
ins_pipe(empty);
|
||||||
|
%}
|
||||||
|
|
||||||
//----------Arithmetic Instructions--------------------------------------------
|
//----------Arithmetic Instructions--------------------------------------------
|
||||||
// Addition Instructions
|
// Addition Instructions
|
||||||
// Register Addition
|
// Register Addition
|
||||||
|
@ -10348,6 +10348,14 @@ instruct castII(iRegIdst dst) %{
|
|||||||
ins_pipe(pipe_class_default);
|
ins_pipe(pipe_class_default);
|
||||||
%}
|
%}
|
||||||
|
|
||||||
|
instruct castLL(iRegLdst dst) %{
|
||||||
|
match(Set dst (CastLL dst));
|
||||||
|
format %{ " -- \t// castLL of $dst" %}
|
||||||
|
size(0);
|
||||||
|
ins_encode( /*empty*/ );
|
||||||
|
ins_pipe(pipe_class_default);
|
||||||
|
%}
|
||||||
|
|
||||||
instruct checkCastPP(iRegPdst dst) %{
|
instruct checkCastPP(iRegPdst dst) %{
|
||||||
match(Set dst (CheckCastPP dst));
|
match(Set dst (CheckCastPP dst));
|
||||||
format %{ " -- \t// checkcastPP of $dst" %}
|
format %{ " -- \t// checkcastPP of $dst" %}
|
||||||
|
@ -5345,6 +5345,13 @@ instruct castII(iRegI dst) %{
|
|||||||
ins_pipe(pipe_class_dummy);
|
ins_pipe(pipe_class_dummy);
|
||||||
%}
|
%}
|
||||||
|
|
||||||
|
instruct castLL(iRegL dst) %{
|
||||||
|
match(Set dst (CastLL dst));
|
||||||
|
size(0);
|
||||||
|
format %{ "# castLL of $dst" %}
|
||||||
|
ins_encode(/*empty*/);
|
||||||
|
ins_pipe(pipe_class_dummy);
|
||||||
|
%}
|
||||||
|
|
||||||
//----------Conditional_store--------------------------------------------------
|
//----------Conditional_store--------------------------------------------------
|
||||||
// Conditional-store of the updated heap-top.
|
// Conditional-store of the updated heap-top.
|
||||||
|
@ -7147,6 +7147,14 @@ instruct castII( rRegI dst ) %{
|
|||||||
ins_pipe( empty );
|
ins_pipe( empty );
|
||||||
%}
|
%}
|
||||||
|
|
||||||
|
instruct castLL( eRegL dst ) %{
|
||||||
|
match(Set dst (CastLL dst));
|
||||||
|
format %{ "#castLL of $dst" %}
|
||||||
|
ins_encode( /*empty encoding*/ );
|
||||||
|
ins_cost(0);
|
||||||
|
ins_pipe( empty );
|
||||||
|
%}
|
||||||
|
|
||||||
// Load-locked - same as a regular pointer load when used with compare-swap
|
// Load-locked - same as a regular pointer load when used with compare-swap
|
||||||
instruct loadPLocked(eRegP dst, memory mem) %{
|
instruct loadPLocked(eRegP dst, memory mem) %{
|
||||||
match(Set dst (LoadPLocked mem));
|
match(Set dst (LoadPLocked mem));
|
||||||
|
@ -7445,6 +7445,17 @@ instruct castII(rRegI dst)
|
|||||||
ins_pipe(empty);
|
ins_pipe(empty);
|
||||||
%}
|
%}
|
||||||
|
|
||||||
|
instruct castLL(rRegL dst)
|
||||||
|
%{
|
||||||
|
match(Set dst (CastLL dst));
|
||||||
|
|
||||||
|
size(0);
|
||||||
|
format %{ "# castLL of $dst" %}
|
||||||
|
ins_encode(/* empty encoding */);
|
||||||
|
ins_cost(0);
|
||||||
|
ins_pipe(empty);
|
||||||
|
%}
|
||||||
|
|
||||||
// LoadP-locked same as a regular LoadP when used with compare-swap
|
// LoadP-locked same as a regular LoadP when used with compare-swap
|
||||||
instruct loadPLocked(rRegP dst, memory mem)
|
instruct loadPLocked(rRegP dst, memory mem)
|
||||||
%{
|
%{
|
||||||
|
@ -337,6 +337,8 @@ class methodHandle;
|
|||||||
\
|
\
|
||||||
do_intrinsic(_Preconditions_checkIndex, jdk_internal_util_Preconditions, checkIndex_name, Preconditions_checkIndex_signature, F_S) \
|
do_intrinsic(_Preconditions_checkIndex, jdk_internal_util_Preconditions, checkIndex_name, Preconditions_checkIndex_signature, F_S) \
|
||||||
do_signature(Preconditions_checkIndex_signature, "(IILjava/util/function/BiFunction;)I") \
|
do_signature(Preconditions_checkIndex_signature, "(IILjava/util/function/BiFunction;)I") \
|
||||||
|
do_intrinsic(_Preconditions_checkLongIndex, jdk_internal_util_Preconditions, checkIndex_name, Preconditions_checkLongIndex_signature, F_S) \
|
||||||
|
do_signature(Preconditions_checkLongIndex_signature, "(JJLjava/util/function/BiFunction;)J") \
|
||||||
\
|
\
|
||||||
do_class(java_nio_Buffer, "java/nio/Buffer") \
|
do_class(java_nio_Buffer, "java/nio/Buffer") \
|
||||||
do_intrinsic(_checkIndex, java_nio_Buffer, checkIndex_name, int_int_signature, F_R) \
|
do_intrinsic(_checkIndex, java_nio_Buffer, checkIndex_name, int_int_signature, F_R) \
|
||||||
|
@ -649,6 +649,7 @@ bool C2Compiler::is_intrinsic_supported(const methodHandle& method, bool is_virt
|
|||||||
case vmIntrinsics::_profileBoolean:
|
case vmIntrinsics::_profileBoolean:
|
||||||
case vmIntrinsics::_isCompileConstant:
|
case vmIntrinsics::_isCompileConstant:
|
||||||
case vmIntrinsics::_Preconditions_checkIndex:
|
case vmIntrinsics::_Preconditions_checkIndex:
|
||||||
|
case vmIntrinsics::_Preconditions_checkLongIndex:
|
||||||
case vmIntrinsics::_getObjectSize:
|
case vmIntrinsics::_getObjectSize:
|
||||||
break;
|
break;
|
||||||
|
|
||||||
|
@ -96,6 +96,11 @@ Node* ConstraintCastNode::make_cast(int opcode, Node* c, Node *n, const Type *t,
|
|||||||
cast->set_req(0, c);
|
cast->set_req(0, c);
|
||||||
return cast;
|
return cast;
|
||||||
}
|
}
|
||||||
|
case Op_CastLL: {
|
||||||
|
Node* cast = new CastLLNode(n, t, carry_dependency);
|
||||||
|
cast->set_req(0, c);
|
||||||
|
return cast;
|
||||||
|
}
|
||||||
case Op_CastPP: {
|
case Op_CastPP: {
|
||||||
Node* cast = new CastPPNode(n, t, carry_dependency);
|
Node* cast = new CastPPNode(n, t, carry_dependency);
|
||||||
cast->set_req(0, c);
|
cast->set_req(0, c);
|
||||||
@ -108,6 +113,20 @@ Node* ConstraintCastNode::make_cast(int opcode, Node* c, Node *n, const Type *t,
|
|||||||
return NULL;
|
return NULL;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
Node* ConstraintCastNode::make(Node* c, Node *n, const Type *t, BasicType bt) {
|
||||||
|
switch(bt) {
|
||||||
|
case T_INT: {
|
||||||
|
return make_cast(Op_CastII, c, n, t, false);
|
||||||
|
}
|
||||||
|
case T_LONG: {
|
||||||
|
return make_cast(Op_CastLL, c, n, t, false);
|
||||||
|
}
|
||||||
|
default:
|
||||||
|
fatal("Bad basic type %s", type2name(bt));
|
||||||
|
}
|
||||||
|
return NULL;
|
||||||
|
}
|
||||||
|
|
||||||
TypeNode* ConstraintCastNode::dominating_cast(PhaseGVN* gvn, PhaseTransform* pt) const {
|
TypeNode* ConstraintCastNode::dominating_cast(PhaseGVN* gvn, PhaseTransform* pt) const {
|
||||||
Node* val = in(1);
|
Node* val = in(1);
|
||||||
Node* ctl = in(0);
|
Node* ctl = in(0);
|
||||||
|
@ -53,6 +53,7 @@ class ConstraintCastNode: public TypeNode {
|
|||||||
bool carry_dependency() const { return _carry_dependency; }
|
bool carry_dependency() const { return _carry_dependency; }
|
||||||
TypeNode* dominating_cast(PhaseGVN* gvn, PhaseTransform* pt) const;
|
TypeNode* dominating_cast(PhaseGVN* gvn, PhaseTransform* pt) const;
|
||||||
static Node* make_cast(int opcode, Node* c, Node *n, const Type *t, bool carry_dependency);
|
static Node* make_cast(int opcode, Node* c, Node *n, const Type *t, bool carry_dependency);
|
||||||
|
static Node* make(Node* c, Node *n, const Type *t, BasicType bt);
|
||||||
|
|
||||||
#ifndef PRODUCT
|
#ifndef PRODUCT
|
||||||
virtual void dump_spec(outputStream *st) const;
|
virtual void dump_spec(outputStream *st) const;
|
||||||
@ -92,6 +93,16 @@ class CastIINode: public ConstraintCastNode {
|
|||||||
#endif
|
#endif
|
||||||
};
|
};
|
||||||
|
|
||||||
|
class CastLLNode: public ConstraintCastNode {
|
||||||
|
public:
|
||||||
|
CastLLNode(Node* n, const Type* t, bool carry_dependency = false)
|
||||||
|
: ConstraintCastNode(n, t, carry_dependency){
|
||||||
|
init_class_id(Class_CastLL);
|
||||||
|
}
|
||||||
|
virtual int Opcode() const;
|
||||||
|
virtual uint ideal_reg() const { return Op_RegL; }
|
||||||
|
};
|
||||||
|
|
||||||
//------------------------------CastPPNode-------------------------------------
|
//------------------------------CastPPNode-------------------------------------
|
||||||
// cast pointer to pointer (different type)
|
// cast pointer to pointer (different type)
|
||||||
class CastPPNode: public ConstraintCastNode {
|
class CastPPNode: public ConstraintCastNode {
|
||||||
|
@ -1970,12 +1970,14 @@ Node *PhiNode::Ideal(PhaseGVN *phase, bool can_reshape) {
|
|||||||
// Wait until after parsing for the type information to propagate from the casts.
|
// Wait until after parsing for the type information to propagate from the casts.
|
||||||
assert(can_reshape, "Invalid during parsing");
|
assert(can_reshape, "Invalid during parsing");
|
||||||
const Type* phi_type = bottom_type();
|
const Type* phi_type = bottom_type();
|
||||||
assert(phi_type->isa_int() || phi_type->isa_ptr(), "bad phi type");
|
assert(phi_type->isa_int() || phi_type->isa_ptr() || phi_type->isa_long(), "bad phi type");
|
||||||
// Add casts to carry the control dependency of the Phi that is
|
// Add casts to carry the control dependency of the Phi that is
|
||||||
// going away
|
// going away
|
||||||
Node* cast = NULL;
|
Node* cast = NULL;
|
||||||
if (phi_type->isa_int()) {
|
if (phi_type->isa_int()) {
|
||||||
cast = ConstraintCastNode::make_cast(Op_CastII, r, uin, phi_type, true);
|
cast = ConstraintCastNode::make_cast(Op_CastII, r, uin, phi_type, true);
|
||||||
|
} else if (phi_type->isa_long()) {
|
||||||
|
cast = ConstraintCastNode::make_cast(Op_CastLL, r, uin, phi_type, true);
|
||||||
} else {
|
} else {
|
||||||
const Type* uin_type = phase->type(uin);
|
const Type* uin_type = phase->type(uin);
|
||||||
if (!phi_type->isa_oopptr() && !uin_type->isa_oopptr()) {
|
if (!phi_type->isa_oopptr() && !uin_type->isa_oopptr()) {
|
||||||
|
@ -61,6 +61,7 @@ macro(CallLeafNoFP)
|
|||||||
macro(CallRuntime)
|
macro(CallRuntime)
|
||||||
macro(CallStaticJava)
|
macro(CallStaticJava)
|
||||||
macro(CastII)
|
macro(CastII)
|
||||||
|
macro(CastLL)
|
||||||
macro(CastX2P)
|
macro(CastX2P)
|
||||||
macro(CastP2X)
|
macro(CastP2X)
|
||||||
macro(CastPP)
|
macro(CastPP)
|
||||||
|
@ -103,6 +103,13 @@ class GraphKit : public Phase {
|
|||||||
// Create or find a constant node
|
// Create or find a constant node
|
||||||
Node* intcon(jint con) const { return _gvn.intcon(con); }
|
Node* intcon(jint con) const { return _gvn.intcon(con); }
|
||||||
Node* longcon(jlong con) const { return _gvn.longcon(con); }
|
Node* longcon(jlong con) const { return _gvn.longcon(con); }
|
||||||
|
Node* integercon(jlong con, BasicType bt) const {
|
||||||
|
if (bt == T_INT) {
|
||||||
|
return intcon(checked_cast<jint>(con));
|
||||||
|
}
|
||||||
|
assert(bt == T_LONG, "basic type not an int or long");
|
||||||
|
return longcon(con);
|
||||||
|
}
|
||||||
Node* makecon(const Type *t) const { return _gvn.makecon(t); }
|
Node* makecon(const Type *t) const { return _gvn.makecon(t); }
|
||||||
Node* zerocon(BasicType bt) const { return _gvn.zerocon(bt); }
|
Node* zerocon(BasicType bt) const { return _gvn.zerocon(bt); }
|
||||||
// (See also macro MakeConX in type.hpp, which uses intcon or longcon.)
|
// (See also macro MakeConX in type.hpp, which uses intcon or longcon.)
|
||||||
|
@ -483,7 +483,8 @@ bool LibraryCallKit::try_to_inline(int predicate) {
|
|||||||
case vmIntrinsics::_copyOfRange: return inline_array_copyOf(true);
|
case vmIntrinsics::_copyOfRange: return inline_array_copyOf(true);
|
||||||
case vmIntrinsics::_equalsB: return inline_array_equals(StrIntrinsicNode::LL);
|
case vmIntrinsics::_equalsB: return inline_array_equals(StrIntrinsicNode::LL);
|
||||||
case vmIntrinsics::_equalsC: return inline_array_equals(StrIntrinsicNode::UU);
|
case vmIntrinsics::_equalsC: return inline_array_equals(StrIntrinsicNode::UU);
|
||||||
case vmIntrinsics::_Preconditions_checkIndex: return inline_preconditions_checkIndex();
|
case vmIntrinsics::_Preconditions_checkIndex: return inline_preconditions_checkIndex(T_INT);
|
||||||
|
case vmIntrinsics::_Preconditions_checkLongIndex: return inline_preconditions_checkIndex(T_LONG);
|
||||||
case vmIntrinsics::_clone: return inline_native_clone(intrinsic()->is_virtual());
|
case vmIntrinsics::_clone: return inline_native_clone(intrinsic()->is_virtual());
|
||||||
|
|
||||||
case vmIntrinsics::_allocateUninitializedArray: return inline_unsafe_newArray(true);
|
case vmIntrinsics::_allocateUninitializedArray: return inline_unsafe_newArray(true);
|
||||||
@ -1001,14 +1002,15 @@ bool LibraryCallKit::inline_hasNegatives() {
|
|||||||
return true;
|
return true;
|
||||||
}
|
}
|
||||||
|
|
||||||
bool LibraryCallKit::inline_preconditions_checkIndex() {
|
bool LibraryCallKit::inline_preconditions_checkIndex(BasicType bt) {
|
||||||
Node* index = argument(0);
|
Node* index = argument(0);
|
||||||
Node* length = argument(1);
|
Node* length = bt == T_INT ? argument(1) : argument(2);
|
||||||
if (too_many_traps(Deoptimization::Reason_intrinsic) || too_many_traps(Deoptimization::Reason_range_check)) {
|
if (too_many_traps(Deoptimization::Reason_intrinsic) || too_many_traps(Deoptimization::Reason_range_check)) {
|
||||||
return false;
|
return false;
|
||||||
}
|
}
|
||||||
|
|
||||||
Node* len_pos_cmp = _gvn.transform(new CmpINode(length, intcon(0)));
|
// check that length is positive
|
||||||
|
Node* len_pos_cmp = _gvn.transform(CmpNode::make(length, integercon(0, bt), bt));
|
||||||
Node* len_pos_bol = _gvn.transform(new BoolNode(len_pos_cmp, BoolTest::ge));
|
Node* len_pos_bol = _gvn.transform(new BoolNode(len_pos_cmp, BoolTest::ge));
|
||||||
|
|
||||||
{
|
{
|
||||||
@ -1017,11 +1019,19 @@ bool LibraryCallKit::inline_preconditions_checkIndex() {
|
|||||||
Deoptimization::Action_make_not_entrant);
|
Deoptimization::Action_make_not_entrant);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// length is now known postive, add a cast node to make this explicit
|
||||||
|
jlong upper_bound = _gvn.type(length)->is_integer(bt)->hi_as_long();
|
||||||
|
Node* casted_length = ConstraintCastNode::make(control(), length, TypeInteger::make(0, upper_bound, Type::WidenMax, bt), bt);
|
||||||
|
casted_length = _gvn.transform(casted_length);
|
||||||
|
replace_in_map(length, casted_length);
|
||||||
|
length = casted_length;
|
||||||
|
|
||||||
if (stopped()) {
|
if (stopped()) {
|
||||||
return false;
|
return false;
|
||||||
}
|
}
|
||||||
|
|
||||||
Node* rc_cmp = _gvn.transform(new CmpUNode(index, length));
|
// Use an unsigned comparison for the range check itself
|
||||||
|
Node* rc_cmp = _gvn.transform(CmpNode::make(index, length, bt, true));
|
||||||
BoolTest::mask btest = BoolTest::lt;
|
BoolTest::mask btest = BoolTest::lt;
|
||||||
Node* rc_bool = _gvn.transform(new BoolNode(rc_cmp, btest));
|
Node* rc_bool = _gvn.transform(new BoolNode(rc_cmp, btest));
|
||||||
RangeCheckNode* rc = new RangeCheckNode(control(), rc_bool, PROB_MAX, COUNT_UNKNOWN);
|
RangeCheckNode* rc = new RangeCheckNode(control(), rc_bool, PROB_MAX, COUNT_UNKNOWN);
|
||||||
@ -1041,8 +1051,8 @@ bool LibraryCallKit::inline_preconditions_checkIndex() {
|
|||||||
return false;
|
return false;
|
||||||
}
|
}
|
||||||
|
|
||||||
Node* result = new CastIINode(index, TypeInt::make(0, _gvn.type(length)->is_int()->_hi, Type::WidenMax));
|
// index is now known to be >= 0 and < length, cast it
|
||||||
result->set_req(0, control());
|
Node* result = ConstraintCastNode::make(control(), index, TypeInteger::make(0, upper_bound, Type::WidenMax, bt), bt);
|
||||||
result = _gvn.transform(result);
|
result = _gvn.transform(result);
|
||||||
set_result(result);
|
set_result(result);
|
||||||
replace_in_map(index, result);
|
replace_in_map(index, result);
|
||||||
|
@ -241,7 +241,7 @@ class LibraryCallKit : public GraphKit {
|
|||||||
bool inline_native_getLength();
|
bool inline_native_getLength();
|
||||||
bool inline_array_copyOf(bool is_copyOfRange);
|
bool inline_array_copyOf(bool is_copyOfRange);
|
||||||
bool inline_array_equals(StrIntrinsicNode::ArgEnc ae);
|
bool inline_array_equals(StrIntrinsicNode::ArgEnc ae);
|
||||||
bool inline_preconditions_checkIndex();
|
bool inline_preconditions_checkIndex(BasicType bt);
|
||||||
void copy_to_clone(Node* obj, Node* alloc_obj, Node* obj_size, bool is_array);
|
void copy_to_clone(Node* obj, Node* alloc_obj, Node* obj_size, bool is_array);
|
||||||
bool inline_native_clone(bool is_virtual);
|
bool inline_native_clone(bool is_virtual);
|
||||||
bool inline_native_Reflection_getCallerClass();
|
bool inline_native_Reflection_getCallerClass();
|
||||||
|
@ -52,6 +52,7 @@ class CallNode;
|
|||||||
class CallRuntimeNode;
|
class CallRuntimeNode;
|
||||||
class CallStaticJavaNode;
|
class CallStaticJavaNode;
|
||||||
class CastIINode;
|
class CastIINode;
|
||||||
|
class CastLLNode;
|
||||||
class CatchNode;
|
class CatchNode;
|
||||||
class CatchProjNode;
|
class CatchProjNode;
|
||||||
class CheckCastPPNode;
|
class CheckCastPPNode;
|
||||||
@ -671,6 +672,7 @@ public:
|
|||||||
DEFINE_CLASS_ID(ConstraintCast, Type, 1)
|
DEFINE_CLASS_ID(ConstraintCast, Type, 1)
|
||||||
DEFINE_CLASS_ID(CastII, ConstraintCast, 0)
|
DEFINE_CLASS_ID(CastII, ConstraintCast, 0)
|
||||||
DEFINE_CLASS_ID(CheckCastPP, ConstraintCast, 1)
|
DEFINE_CLASS_ID(CheckCastPP, ConstraintCast, 1)
|
||||||
|
DEFINE_CLASS_ID(CastLL, ConstraintCast, 2)
|
||||||
DEFINE_CLASS_ID(CMove, Type, 3)
|
DEFINE_CLASS_ID(CMove, Type, 3)
|
||||||
DEFINE_CLASS_ID(SafePointScalarObject, Type, 4)
|
DEFINE_CLASS_ID(SafePointScalarObject, Type, 4)
|
||||||
DEFINE_CLASS_ID(DecodeNarrowPtr, Type, 5)
|
DEFINE_CLASS_ID(DecodeNarrowPtr, Type, 5)
|
||||||
@ -819,6 +821,7 @@ public:
|
|||||||
DEFINE_CLASS_QUERY(CatchProj)
|
DEFINE_CLASS_QUERY(CatchProj)
|
||||||
DEFINE_CLASS_QUERY(CheckCastPP)
|
DEFINE_CLASS_QUERY(CheckCastPP)
|
||||||
DEFINE_CLASS_QUERY(CastII)
|
DEFINE_CLASS_QUERY(CastII)
|
||||||
|
DEFINE_CLASS_QUERY(CastLL)
|
||||||
DEFINE_CLASS_QUERY(ConstraintCast)
|
DEFINE_CLASS_QUERY(ConstraintCast)
|
||||||
DEFINE_CLASS_QUERY(ClearArray)
|
DEFINE_CLASS_QUERY(ClearArray)
|
||||||
DEFINE_CLASS_QUERY(CMove)
|
DEFINE_CLASS_QUERY(CMove)
|
||||||
|
@ -274,6 +274,7 @@ public:
|
|||||||
// Fast int or long constant. Same as TypeInt::make(i) or TypeLong::make(l).
|
// Fast int or long constant. Same as TypeInt::make(i) or TypeLong::make(l).
|
||||||
ConINode* intcon(jint i);
|
ConINode* intcon(jint i);
|
||||||
ConLNode* longcon(jlong l);
|
ConLNode* longcon(jlong l);
|
||||||
|
ConNode* integercon(jlong l, BasicType bt);
|
||||||
|
|
||||||
// Fast zero or null constant. Same as makecon(Type::get_zero_type(bt)).
|
// Fast zero or null constant. Same as makecon(Type::get_zero_type(bt)).
|
||||||
ConNode* zerocon(BasicType bt);
|
ConNode* zerocon(BasicType bt);
|
||||||
|
@ -563,8 +563,27 @@ void CmpNode::related(GrowableArray<Node*> *in_rel, GrowableArray<Node*> *out_re
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
#endif
|
#endif
|
||||||
|
|
||||||
|
CmpNode *CmpNode::make(Node *in1, Node *in2, BasicType bt, bool unsigned_comp) {
|
||||||
|
switch (bt) {
|
||||||
|
case T_INT:
|
||||||
|
if (unsigned_comp) {
|
||||||
|
return new CmpUNode(in1, in2);
|
||||||
|
}
|
||||||
|
return new CmpINode(in1, in2);
|
||||||
|
case T_LONG:
|
||||||
|
if (unsigned_comp) {
|
||||||
|
return new CmpULNode(in1, in2);
|
||||||
|
}
|
||||||
|
return new CmpLNode(in1, in2);
|
||||||
|
default:
|
||||||
|
fatal("Not implemented for %s", type2name(bt));
|
||||||
|
}
|
||||||
|
return NULL;
|
||||||
|
}
|
||||||
|
|
||||||
//=============================================================================
|
//=============================================================================
|
||||||
//------------------------------cmp--------------------------------------------
|
//------------------------------cmp--------------------------------------------
|
||||||
// Simplify a CmpI (compare 2 integers) node, based on local information.
|
// Simplify a CmpI (compare 2 integers) node, based on local information.
|
||||||
|
@ -140,6 +140,8 @@ public:
|
|||||||
const Type *bottom_type() const { return TypeInt::CC; }
|
const Type *bottom_type() const { return TypeInt::CC; }
|
||||||
virtual uint ideal_reg() const { return Op_RegFlags; }
|
virtual uint ideal_reg() const { return Op_RegFlags; }
|
||||||
|
|
||||||
|
static CmpNode *make(Node *in1, Node *in2, BasicType bt, bool unsigned_comp = false);
|
||||||
|
|
||||||
#ifndef PRODUCT
|
#ifndef PRODUCT
|
||||||
// CmpNode and subclasses include all data inputs (until hitting a control
|
// CmpNode and subclasses include all data inputs (until hitting a control
|
||||||
// boundary) in their related node set, as well as all outputs until and
|
// boundary) in their related node set, as well as all outputs until and
|
||||||
|
@ -1338,6 +1338,14 @@ bool TypeD::empty(void) const {
|
|||||||
return false; // always exactly a singleton
|
return false; // always exactly a singleton
|
||||||
}
|
}
|
||||||
|
|
||||||
|
const TypeInteger* TypeInteger::make(jlong lo, jlong hi, int w, BasicType bt) {
|
||||||
|
if (bt == T_INT) {
|
||||||
|
return TypeInt::make(checked_cast<jint>(lo), checked_cast<jint>(hi), w);
|
||||||
|
}
|
||||||
|
assert(bt == T_LONG, "basic type not an int or long");
|
||||||
|
return TypeLong::make(lo, hi, w);
|
||||||
|
}
|
||||||
|
|
||||||
//=============================================================================
|
//=============================================================================
|
||||||
// Convience common pre-built types.
|
// Convience common pre-built types.
|
||||||
const TypeInt *TypeInt::MAX; // INT_MAX
|
const TypeInt *TypeInt::MAX; // INT_MAX
|
||||||
@ -1363,7 +1371,7 @@ const TypeInt *TypeInt::SYMINT; // symmetric range [-max_jint..max_jint]
|
|||||||
const TypeInt *TypeInt::TYPE_DOMAIN; // alias for TypeInt::INT
|
const TypeInt *TypeInt::TYPE_DOMAIN; // alias for TypeInt::INT
|
||||||
|
|
||||||
//------------------------------TypeInt----------------------------------------
|
//------------------------------TypeInt----------------------------------------
|
||||||
TypeInt::TypeInt( jint lo, jint hi, int w ) : Type(Int), _lo(lo), _hi(hi), _widen(w) {
|
TypeInt::TypeInt( jint lo, jint hi, int w ) : TypeInteger(Int), _lo(lo), _hi(hi), _widen(w) {
|
||||||
}
|
}
|
||||||
|
|
||||||
//------------------------------make-------------------------------------------
|
//------------------------------make-------------------------------------------
|
||||||
@ -1623,7 +1631,7 @@ const TypeLong *TypeLong::UINT; // 32-bit unsigned subrange
|
|||||||
const TypeLong *TypeLong::TYPE_DOMAIN; // alias for TypeLong::LONG
|
const TypeLong *TypeLong::TYPE_DOMAIN; // alias for TypeLong::LONG
|
||||||
|
|
||||||
//------------------------------TypeLong---------------------------------------
|
//------------------------------TypeLong---------------------------------------
|
||||||
TypeLong::TypeLong( jlong lo, jlong hi, int w ) : Type(Long), _lo(lo), _hi(hi), _widen(w) {
|
TypeLong::TypeLong(jlong lo, jlong hi, int w) : TypeInteger(Long), _lo(lo), _hi(hi), _widen(w) {
|
||||||
}
|
}
|
||||||
|
|
||||||
//------------------------------make-------------------------------------------
|
//------------------------------make-------------------------------------------
|
||||||
|
@ -45,8 +45,9 @@ class Dict;
|
|||||||
class Type;
|
class Type;
|
||||||
class TypeD;
|
class TypeD;
|
||||||
class TypeF;
|
class TypeF;
|
||||||
class TypeInt;
|
class TypeInteger;
|
||||||
class TypeLong;
|
class TypeInt;
|
||||||
|
class TypeLong;
|
||||||
class TypeNarrowPtr;
|
class TypeNarrowPtr;
|
||||||
class TypeNarrowOop;
|
class TypeNarrowOop;
|
||||||
class TypeNarrowKlass;
|
class TypeNarrowKlass;
|
||||||
@ -283,6 +284,9 @@ public:
|
|||||||
|
|
||||||
const TypeInt *is_int() const;
|
const TypeInt *is_int() const;
|
||||||
const TypeInt *isa_int() const; // Returns NULL if not an Int
|
const TypeInt *isa_int() const; // Returns NULL if not an Int
|
||||||
|
const TypeInteger* isa_integer() const;
|
||||||
|
const TypeInteger* is_integer(BasicType bt) const;
|
||||||
|
const TypeInteger* isa_integer(BasicType bt) const;
|
||||||
const TypeLong *is_long() const;
|
const TypeLong *is_long() const;
|
||||||
const TypeLong *isa_long() const; // Returns NULL if not a Long
|
const TypeLong *isa_long() const; // Returns NULL if not a Long
|
||||||
const TypeD *isa_double() const; // Returns NULL if not a Double{Top,Con,Bot}
|
const TypeD *isa_double() const; // Returns NULL if not a Double{Top,Con,Bot}
|
||||||
@ -525,10 +529,24 @@ public:
|
|||||||
#endif
|
#endif
|
||||||
};
|
};
|
||||||
|
|
||||||
|
class TypeInteger : public Type {
|
||||||
|
protected:
|
||||||
|
TypeInteger(TYPES t) : Type(t) {}
|
||||||
|
|
||||||
|
public:
|
||||||
|
virtual jlong hi_as_long() const = 0;
|
||||||
|
virtual jlong lo_as_long() const = 0;
|
||||||
|
jlong get_con_as_long(BasicType bt) const;
|
||||||
|
|
||||||
|
static const TypeInteger* make(jlong lo, jlong hi, int w, BasicType bt);
|
||||||
|
};
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
//------------------------------TypeInt----------------------------------------
|
//------------------------------TypeInt----------------------------------------
|
||||||
// Class of integer ranges, the set of integers between a lower bound and an
|
// Class of integer ranges, the set of integers between a lower bound and an
|
||||||
// upper bound, inclusive.
|
// upper bound, inclusive.
|
||||||
class TypeInt : public Type {
|
class TypeInt : public TypeInteger {
|
||||||
TypeInt( jint lo, jint hi, int w );
|
TypeInt( jint lo, jint hi, int w );
|
||||||
protected:
|
protected:
|
||||||
virtual const Type *filter_helper(const Type *kills, bool include_speculative) const;
|
virtual const Type *filter_helper(const Type *kills, bool include_speculative) const;
|
||||||
@ -557,6 +575,10 @@ public:
|
|||||||
virtual const Type *xdual() const; // Compute dual right now.
|
virtual const Type *xdual() const; // Compute dual right now.
|
||||||
virtual const Type *widen( const Type *t, const Type* limit_type ) const;
|
virtual const Type *widen( const Type *t, const Type* limit_type ) const;
|
||||||
virtual const Type *narrow( const Type *t ) const;
|
virtual const Type *narrow( const Type *t ) const;
|
||||||
|
|
||||||
|
virtual jlong hi_as_long() const { return _hi; }
|
||||||
|
virtual jlong lo_as_long() const { return _lo; }
|
||||||
|
|
||||||
// Do not kill _widen bits.
|
// Do not kill _widen bits.
|
||||||
// Convenience common pre-built types.
|
// Convenience common pre-built types.
|
||||||
static const TypeInt *MAX;
|
static const TypeInt *MAX;
|
||||||
@ -591,7 +613,7 @@ public:
|
|||||||
//------------------------------TypeLong---------------------------------------
|
//------------------------------TypeLong---------------------------------------
|
||||||
// Class of long integer ranges, the set of integers between a lower bound and
|
// Class of long integer ranges, the set of integers between a lower bound and
|
||||||
// an upper bound, inclusive.
|
// an upper bound, inclusive.
|
||||||
class TypeLong : public Type {
|
class TypeLong : public TypeInteger {
|
||||||
TypeLong( jlong lo, jlong hi, int w );
|
TypeLong( jlong lo, jlong hi, int w );
|
||||||
protected:
|
protected:
|
||||||
// Do not kill _widen bits.
|
// Do not kill _widen bits.
|
||||||
@ -620,6 +642,8 @@ public:
|
|||||||
|
|
||||||
virtual bool is_finite() const; // Has a finite value
|
virtual bool is_finite() const; // Has a finite value
|
||||||
|
|
||||||
|
virtual jlong hi_as_long() const { return _hi; }
|
||||||
|
virtual jlong lo_as_long() const { return _lo; }
|
||||||
|
|
||||||
virtual const Type *xmeet( const Type *t ) const;
|
virtual const Type *xmeet( const Type *t ) const;
|
||||||
virtual const Type *xdual() const; // Compute dual right now.
|
virtual const Type *xdual() const; // Compute dual right now.
|
||||||
@ -1575,6 +1599,15 @@ inline double Type::getd() const {
|
|||||||
return ((TypeD*)this)->_d;
|
return ((TypeD*)this)->_d;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
inline const TypeInteger *Type::is_integer(BasicType bt) const {
|
||||||
|
assert((bt == T_INT && _base == Int) || (bt == T_LONG && _base == Long), "Not an Int");
|
||||||
|
return (TypeInteger*)this;
|
||||||
|
}
|
||||||
|
|
||||||
|
inline const TypeInteger *Type::isa_integer(BasicType bt) const {
|
||||||
|
return (((bt == T_INT && _base == Int) || (bt == T_LONG && _base == Long)) ? (TypeInteger*)this : NULL);
|
||||||
|
}
|
||||||
|
|
||||||
inline const TypeInt *Type::is_int() const {
|
inline const TypeInt *Type::is_int() const {
|
||||||
assert( _base == Int, "Not an Int" );
|
assert( _base == Int, "Not an Int" );
|
||||||
return (TypeInt*)this;
|
return (TypeInt*)this;
|
||||||
|
@ -1,5 +1,5 @@
|
|||||||
/*
|
/*
|
||||||
* Copyright (c) 1995, 2019, Oracle and/or its affiliates. All rights reserved.
|
* Copyright (c) 1995, 2020, Oracle and/or its affiliates. All rights reserved.
|
||||||
* DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER.
|
* DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER.
|
||||||
*
|
*
|
||||||
* This code is free software; you can redistribute it and/or modify it
|
* This code is free software; you can redistribute it and/or modify it
|
||||||
@ -68,4 +68,18 @@ public class IndexOutOfBoundsException extends RuntimeException {
|
|||||||
public IndexOutOfBoundsException(int index) {
|
public IndexOutOfBoundsException(int index) {
|
||||||
super("Index out of range: " + index);
|
super("Index out of range: " + index);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Constructs a new {@code IndexOutOfBoundsException} class with an
|
||||||
|
* argument indicating the illegal index.
|
||||||
|
*
|
||||||
|
* <p>The index is included in this exception's detail message. The
|
||||||
|
* exact presentation format of the detail message is unspecified.
|
||||||
|
*
|
||||||
|
* @param index the illegal index.
|
||||||
|
* @since 16
|
||||||
|
*/
|
||||||
|
public IndexOutOfBoundsException(long index) {
|
||||||
|
super("Index out of range: " + index);
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
@ -2146,7 +2146,7 @@ public abstract class VarHandle implements Constable {
|
|||||||
UNSAFE.fullFence();
|
UNSAFE.fullFence();
|
||||||
}
|
}
|
||||||
|
|
||||||
static final BiFunction<String, List<Integer>, ArrayIndexOutOfBoundsException>
|
static final BiFunction<String, List<Number>, ArrayIndexOutOfBoundsException>
|
||||||
AIOOBE_SUPPLIER = Preconditions.outOfBoundsExceptionFormatter(
|
AIOOBE_SUPPLIER = Preconditions.outOfBoundsExceptionFormatter(
|
||||||
new Function<String, ArrayIndexOutOfBoundsException>() {
|
new Function<String, ArrayIndexOutOfBoundsException>() {
|
||||||
@Override
|
@Override
|
||||||
|
@ -411,4 +411,79 @@ public final class Objects {
|
|||||||
return Preconditions.checkFromIndexSize(fromIndex, size, length, null);
|
return Preconditions.checkFromIndexSize(fromIndex, size, length, null);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Checks if the {@code index} is within the bounds of the range from
|
||||||
|
* {@code 0} (inclusive) to {@code length} (exclusive).
|
||||||
|
*
|
||||||
|
* <p>The {@code index} is defined to be out of bounds if any of the
|
||||||
|
* following inequalities is true:
|
||||||
|
* <ul>
|
||||||
|
* <li>{@code index < 0}</li>
|
||||||
|
* <li>{@code index >= length}</li>
|
||||||
|
* <li>{@code length < 0}, which is implied from the former inequalities</li>
|
||||||
|
* </ul>
|
||||||
|
*
|
||||||
|
* @param index the index
|
||||||
|
* @param length the upper-bound (exclusive) of the range
|
||||||
|
* @return {@code index} if it is within bounds of the range
|
||||||
|
* @throws IndexOutOfBoundsException if the {@code index} is out of bounds
|
||||||
|
* @since 16
|
||||||
|
*/
|
||||||
|
@ForceInline
|
||||||
|
public static
|
||||||
|
long checkIndex(long index, long length) {
|
||||||
|
return Preconditions.checkIndex(index, length, null);
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Checks if the sub-range from {@code fromIndex} (inclusive) to
|
||||||
|
* {@code toIndex} (exclusive) is within the bounds of range from {@code 0}
|
||||||
|
* (inclusive) to {@code length} (exclusive).
|
||||||
|
*
|
||||||
|
* <p>The sub-range is defined to be out of bounds if any of the following
|
||||||
|
* inequalities is true:
|
||||||
|
* <ul>
|
||||||
|
* <li>{@code fromIndex < 0}</li>
|
||||||
|
* <li>{@code fromIndex > toIndex}</li>
|
||||||
|
* <li>{@code toIndex > length}</li>
|
||||||
|
* <li>{@code length < 0}, which is implied from the former inequalities</li>
|
||||||
|
* </ul>
|
||||||
|
*
|
||||||
|
* @param fromIndex the lower-bound (inclusive) of the sub-range
|
||||||
|
* @param toIndex the upper-bound (exclusive) of the sub-range
|
||||||
|
* @param length the upper-bound (exclusive) the range
|
||||||
|
* @return {@code fromIndex} if the sub-range within bounds of the range
|
||||||
|
* @throws IndexOutOfBoundsException if the sub-range is out of bounds
|
||||||
|
* @since 16
|
||||||
|
*/
|
||||||
|
public static
|
||||||
|
long checkFromToIndex(long fromIndex, long toIndex, long length) {
|
||||||
|
return Preconditions.checkFromToIndex(fromIndex, toIndex, length, null);
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Checks if the sub-range from {@code fromIndex} (inclusive) to
|
||||||
|
* {@code fromIndex + size} (exclusive) is within the bounds of range from
|
||||||
|
* {@code 0} (inclusive) to {@code length} (exclusive).
|
||||||
|
*
|
||||||
|
* <p>The sub-range is defined to be out of bounds if any of the following
|
||||||
|
* inequalities is true:
|
||||||
|
* <ul>
|
||||||
|
* <li>{@code fromIndex < 0}</li>
|
||||||
|
* <li>{@code size < 0}</li>
|
||||||
|
* <li>{@code fromIndex + size > length}, taking into account integer overflow</li>
|
||||||
|
* <li>{@code length < 0}, which is implied from the former inequalities</li>
|
||||||
|
* </ul>
|
||||||
|
*
|
||||||
|
* @param fromIndex the lower-bound (inclusive) of the sub-interval
|
||||||
|
* @param size the size of the sub-range
|
||||||
|
* @param length the upper-bound (exclusive) of the range
|
||||||
|
* @return {@code fromIndex} if the sub-range within bounds of the range
|
||||||
|
* @throws IndexOutOfBoundsException if the sub-range is out of bounds
|
||||||
|
* @since 16
|
||||||
|
*/
|
||||||
|
public static
|
||||||
|
long checkFromIndexSize(long fromIndex, long size, long length) {
|
||||||
|
return Preconditions.checkFromIndexSize(fromIndex, size, length, null);
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
@ -54,10 +54,10 @@ public class Preconditions {
|
|||||||
* @return the runtime exception
|
* @return the runtime exception
|
||||||
*/
|
*/
|
||||||
private static RuntimeException outOfBounds(
|
private static RuntimeException outOfBounds(
|
||||||
BiFunction<String, List<Integer>, ? extends RuntimeException> oobef,
|
BiFunction<String, List<Number>, ? extends RuntimeException> oobef,
|
||||||
String checkKind,
|
String checkKind,
|
||||||
Integer... args) {
|
Number... args) {
|
||||||
List<Integer> largs = List.of(args);
|
List<Number> largs = List.of(args);
|
||||||
RuntimeException e = oobef == null
|
RuntimeException e = oobef == null
|
||||||
? null : oobef.apply(checkKind, largs);
|
? null : oobef.apply(checkKind, largs);
|
||||||
return e == null
|
return e == null
|
||||||
@ -65,23 +65,41 @@ public class Preconditions {
|
|||||||
}
|
}
|
||||||
|
|
||||||
private static RuntimeException outOfBoundsCheckIndex(
|
private static RuntimeException outOfBoundsCheckIndex(
|
||||||
BiFunction<String, List<Integer>, ? extends RuntimeException> oobe,
|
BiFunction<String, List<Number>, ? extends RuntimeException> oobe,
|
||||||
int index, int length) {
|
int index, int length) {
|
||||||
return outOfBounds(oobe, "checkIndex", index, length);
|
return outOfBounds(oobe, "checkIndex", index, length);
|
||||||
}
|
}
|
||||||
|
|
||||||
private static RuntimeException outOfBoundsCheckFromToIndex(
|
private static RuntimeException outOfBoundsCheckFromToIndex(
|
||||||
BiFunction<String, List<Integer>, ? extends RuntimeException> oobe,
|
BiFunction<String, List<Number>, ? extends RuntimeException> oobe,
|
||||||
int fromIndex, int toIndex, int length) {
|
int fromIndex, int toIndex, int length) {
|
||||||
return outOfBounds(oobe, "checkFromToIndex", fromIndex, toIndex, length);
|
return outOfBounds(oobe, "checkFromToIndex", fromIndex, toIndex, length);
|
||||||
}
|
}
|
||||||
|
|
||||||
private static RuntimeException outOfBoundsCheckFromIndexSize(
|
private static RuntimeException outOfBoundsCheckFromIndexSize(
|
||||||
BiFunction<String, List<Integer>, ? extends RuntimeException> oobe,
|
BiFunction<String, List<Number>, ? extends RuntimeException> oobe,
|
||||||
int fromIndex, int size, int length) {
|
int fromIndex, int size, int length) {
|
||||||
return outOfBounds(oobe, "checkFromIndexSize", fromIndex, size, length);
|
return outOfBounds(oobe, "checkFromIndexSize", fromIndex, size, length);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
private static RuntimeException outOfBoundsCheckIndex(
|
||||||
|
BiFunction<String, List<Number>, ? extends RuntimeException> oobe,
|
||||||
|
long index, long length) {
|
||||||
|
return outOfBounds(oobe, "checkIndex", index, length);
|
||||||
|
}
|
||||||
|
|
||||||
|
private static RuntimeException outOfBoundsCheckFromToIndex(
|
||||||
|
BiFunction<String, List<Number>, ? extends RuntimeException> oobe,
|
||||||
|
long fromIndex, long toIndex, long length) {
|
||||||
|
return outOfBounds(oobe, "checkFromToIndex", fromIndex, toIndex, length);
|
||||||
|
}
|
||||||
|
|
||||||
|
private static RuntimeException outOfBoundsCheckFromIndexSize(
|
||||||
|
BiFunction<String, List<Number>, ? extends RuntimeException> oobe,
|
||||||
|
long fromIndex, long size, long length) {
|
||||||
|
return outOfBounds(oobe, "checkFromIndexSize", fromIndex, size, length);
|
||||||
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Returns an out-of-bounds exception formatter from an given exception
|
* Returns an out-of-bounds exception formatter from an given exception
|
||||||
* factory. The exception formatter is a function that formats an
|
* factory. The exception formatter is a function that formats an
|
||||||
@ -90,8 +108,8 @@ public class Preconditions {
|
|||||||
*
|
*
|
||||||
* <p>The exception formatter accepts two arguments: a {@code String}
|
* <p>The exception formatter accepts two arguments: a {@code String}
|
||||||
* describing the out-of-bounds range check that failed, referred to as the
|
* describing the out-of-bounds range check that failed, referred to as the
|
||||||
* <em>check kind</em>; and a {@code List<Integer>} containing the
|
* <em>check kind</em>; and a {@code List<Number>} containing the
|
||||||
* out-of-bound integer values that failed the check. The list of
|
* out-of-bound integral values that failed the check. The list of
|
||||||
* out-of-bound values is not modified.
|
* out-of-bound values is not modified.
|
||||||
*
|
*
|
||||||
* <p>Three check kinds are supported {@code checkIndex},
|
* <p>Three check kinds are supported {@code checkIndex},
|
||||||
@ -102,7 +120,7 @@ public class Preconditions {
|
|||||||
* {@link #checkFromToIndex(int, int, int, BiFunction) checkFromToIndex}, and
|
* {@link #checkFromToIndex(int, int, int, BiFunction) checkFromToIndex}, and
|
||||||
* {@link #checkFromIndexSize(int, int, int, BiFunction) checkFromIndexSize}.
|
* {@link #checkFromIndexSize(int, int, int, BiFunction) checkFromIndexSize}.
|
||||||
* Thus a supported check kind corresponds to a method name and the
|
* Thus a supported check kind corresponds to a method name and the
|
||||||
* out-of-bound integer values correspond to method argument values, in
|
* out-of-bound integral values correspond to method argument values, in
|
||||||
* order, preceding the exception formatter argument (similar in many
|
* order, preceding the exception formatter argument (similar in many
|
||||||
* respects to the form of arguments required for a reflective invocation of
|
* respects to the form of arguments required for a reflective invocation of
|
||||||
* such a range check method).
|
* such a range check method).
|
||||||
@ -123,7 +141,7 @@ public class Preconditions {
|
|||||||
* {@code static final} field as follows:
|
* {@code static final} field as follows:
|
||||||
* <pre>{@code
|
* <pre>{@code
|
||||||
* static final
|
* static final
|
||||||
* BiFunction<String, List<Integer>, ArrayIndexOutOfBoundsException> AIOOBEF =
|
* BiFunction<String, List<Number>, ArrayIndexOutOfBoundsException> AIOOBEF =
|
||||||
* outOfBoundsExceptionFormatter(ArrayIndexOutOfBoundsException::new);
|
* outOfBoundsExceptionFormatter(ArrayIndexOutOfBoundsException::new);
|
||||||
* }</pre>
|
* }</pre>
|
||||||
* The formatter instance {@code AIOOBEF} may be passed as an argument to an
|
* The formatter instance {@code AIOOBEF} may be passed as an argument to an
|
||||||
@ -150,18 +168,18 @@ public class Preconditions {
|
|||||||
* @return the out-of-bounds exception formatter
|
* @return the out-of-bounds exception formatter
|
||||||
*/
|
*/
|
||||||
public static <X extends RuntimeException>
|
public static <X extends RuntimeException>
|
||||||
BiFunction<String, List<Integer>, X> outOfBoundsExceptionFormatter(Function<String, X> f) {
|
BiFunction<String, List<Number>, X> outOfBoundsExceptionFormatter(Function<String, X> f) {
|
||||||
// Use anonymous class to avoid bootstrap issues if this method is
|
// Use anonymous class to avoid bootstrap issues if this method is
|
||||||
// used early in startup
|
// used early in startup
|
||||||
return new BiFunction<String, List<Integer>, X>() {
|
return new BiFunction<String, List<Number>, X>() {
|
||||||
@Override
|
@Override
|
||||||
public X apply(String checkKind, List<Integer> args) {
|
public X apply(String checkKind, List<Number> args) {
|
||||||
return f.apply(outOfBoundsMessage(checkKind, args));
|
return f.apply(outOfBoundsMessage(checkKind, args));
|
||||||
}
|
}
|
||||||
};
|
};
|
||||||
}
|
}
|
||||||
|
|
||||||
private static String outOfBoundsMessage(String checkKind, List<Integer> args) {
|
private static String outOfBoundsMessage(String checkKind, List<? extends Number> args) {
|
||||||
if (checkKind == null && args == null) {
|
if (checkKind == null && args == null) {
|
||||||
return String.format("Range check failed");
|
return String.format("Range check failed");
|
||||||
} else if (checkKind == null) {
|
} else if (checkKind == null) {
|
||||||
@ -213,7 +231,7 @@ public class Preconditions {
|
|||||||
* <p>If the {@code index} is out of bounds, then a runtime exception is
|
* <p>If the {@code index} is out of bounds, then a runtime exception is
|
||||||
* thrown that is the result of applying the following arguments to the
|
* thrown that is the result of applying the following arguments to the
|
||||||
* exception formatter: the name of this method, {@code checkIndex};
|
* exception formatter: the name of this method, {@code checkIndex};
|
||||||
* and an unmodifiable list integers whose values are, in order, the
|
* and an unmodifiable list of integers whose values are, in order, the
|
||||||
* out-of-bounds arguments {@code index} and {@code length}.
|
* out-of-bounds arguments {@code index} and {@code length}.
|
||||||
*
|
*
|
||||||
* @param <X> the type of runtime exception to throw if the arguments are
|
* @param <X> the type of runtime exception to throw if the arguments are
|
||||||
@ -243,7 +261,7 @@ public class Preconditions {
|
|||||||
@IntrinsicCandidate
|
@IntrinsicCandidate
|
||||||
public static <X extends RuntimeException>
|
public static <X extends RuntimeException>
|
||||||
int checkIndex(int index, int length,
|
int checkIndex(int index, int length,
|
||||||
BiFunction<String, List<Integer>, X> oobef) {
|
BiFunction<String, List<Number>, X> oobef) {
|
||||||
if (index < 0 || index >= length)
|
if (index < 0 || index >= length)
|
||||||
throw outOfBoundsCheckIndex(oobef, index, length);
|
throw outOfBoundsCheckIndex(oobef, index, length);
|
||||||
return index;
|
return index;
|
||||||
@ -266,7 +284,7 @@ public class Preconditions {
|
|||||||
* <p>If the sub-range is out of bounds, then a runtime exception is
|
* <p>If the sub-range is out of bounds, then a runtime exception is
|
||||||
* thrown that is the result of applying the following arguments to the
|
* thrown that is the result of applying the following arguments to the
|
||||||
* exception formatter: the name of this method, {@code checkFromToIndex};
|
* exception formatter: the name of this method, {@code checkFromToIndex};
|
||||||
* and an unmodifiable list integers whose values are, in order, the
|
* and an unmodifiable list of integers whose values are, in order, the
|
||||||
* out-of-bounds arguments {@code fromIndex}, {@code toIndex}, and {@code length}.
|
* out-of-bounds arguments {@code fromIndex}, {@code toIndex}, and {@code length}.
|
||||||
*
|
*
|
||||||
* @param <X> the type of runtime exception to throw if the arguments are
|
* @param <X> the type of runtime exception to throw if the arguments are
|
||||||
@ -290,7 +308,7 @@ public class Preconditions {
|
|||||||
*/
|
*/
|
||||||
public static <X extends RuntimeException>
|
public static <X extends RuntimeException>
|
||||||
int checkFromToIndex(int fromIndex, int toIndex, int length,
|
int checkFromToIndex(int fromIndex, int toIndex, int length,
|
||||||
BiFunction<String, List<Integer>, X> oobef) {
|
BiFunction<String, List<Number>, X> oobef) {
|
||||||
if (fromIndex < 0 || fromIndex > toIndex || toIndex > length)
|
if (fromIndex < 0 || fromIndex > toIndex || toIndex > length)
|
||||||
throw outOfBoundsCheckFromToIndex(oobef, fromIndex, toIndex, length);
|
throw outOfBoundsCheckFromToIndex(oobef, fromIndex, toIndex, length);
|
||||||
return fromIndex;
|
return fromIndex;
|
||||||
@ -313,7 +331,7 @@ public class Preconditions {
|
|||||||
* <p>If the sub-range is out of bounds, then a runtime exception is
|
* <p>If the sub-range is out of bounds, then a runtime exception is
|
||||||
* thrown that is the result of applying the following arguments to the
|
* thrown that is the result of applying the following arguments to the
|
||||||
* exception formatter: the name of this method, {@code checkFromIndexSize};
|
* exception formatter: the name of this method, {@code checkFromIndexSize};
|
||||||
* and an unmodifiable list integers whose values are, in order, the
|
* and an unmodifiable list of integers whose values are, in order, the
|
||||||
* out-of-bounds arguments {@code fromIndex}, {@code size}, and
|
* out-of-bounds arguments {@code fromIndex}, {@code size}, and
|
||||||
* {@code length}.
|
* {@code length}.
|
||||||
*
|
*
|
||||||
@ -338,7 +356,153 @@ public class Preconditions {
|
|||||||
*/
|
*/
|
||||||
public static <X extends RuntimeException>
|
public static <X extends RuntimeException>
|
||||||
int checkFromIndexSize(int fromIndex, int size, int length,
|
int checkFromIndexSize(int fromIndex, int size, int length,
|
||||||
BiFunction<String, List<Integer>, X> oobef) {
|
BiFunction<String, List<Number>, X> oobef) {
|
||||||
|
if ((length | fromIndex | size) < 0 || size > length - fromIndex)
|
||||||
|
throw outOfBoundsCheckFromIndexSize(oobef, fromIndex, size, length);
|
||||||
|
return fromIndex;
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Checks if the {@code index} is within the bounds of the range from
|
||||||
|
* {@code 0} (inclusive) to {@code length} (exclusive).
|
||||||
|
*
|
||||||
|
* <p>The {@code index} is defined to be out of bounds if any of the
|
||||||
|
* following inequalities is true:
|
||||||
|
* <ul>
|
||||||
|
* <li>{@code index < 0}</li>
|
||||||
|
* <li>{@code index >= length}</li>
|
||||||
|
* <li>{@code length < 0}, which is implied from the former inequalities</li>
|
||||||
|
* </ul>
|
||||||
|
*
|
||||||
|
* <p>If the {@code index} is out of bounds, then a runtime exception is
|
||||||
|
* thrown that is the result of applying the following arguments to the
|
||||||
|
* exception formatter: the name of this method, {@code checkIndex};
|
||||||
|
* and an unmodifiable list of longs whose values are, in order, the
|
||||||
|
* out-of-bounds arguments {@code index} and {@code length}.
|
||||||
|
*
|
||||||
|
* @param <X> the type of runtime exception to throw if the arguments are
|
||||||
|
* out of bounds
|
||||||
|
* @param index the index
|
||||||
|
* @param length the upper-bound (exclusive) of the range
|
||||||
|
* @param oobef the exception formatter that when applied with this
|
||||||
|
* method name and out-of-bounds arguments returns a runtime
|
||||||
|
* exception. If {@code null} or returns {@code null} then, it is as
|
||||||
|
* if an exception formatter produced from an invocation of
|
||||||
|
* {@code outOfBoundsExceptionFormatter(IndexOutOfBounds::new)} is used
|
||||||
|
* instead (though it may be more efficient).
|
||||||
|
* Exceptions thrown by the formatter are relayed to the caller.
|
||||||
|
* @return {@code index} if it is within bounds of the range
|
||||||
|
* @throws X if the {@code index} is out of bounds and the exception
|
||||||
|
* formatter is non-{@code null}
|
||||||
|
* @throws IndexOutOfBoundsException if the {@code index} is out of bounds
|
||||||
|
* and the exception formatter is {@code null}
|
||||||
|
* @since 16
|
||||||
|
*
|
||||||
|
* @implNote
|
||||||
|
* This method is made intrinsic in optimizing compilers to guide them to
|
||||||
|
* perform unsigned comparisons of the index and length when it is known the
|
||||||
|
* length is a non-negative value (such as that of an array length or from
|
||||||
|
* the upper bound of a loop)
|
||||||
|
*/
|
||||||
|
@IntrinsicCandidate
|
||||||
|
public static <X extends RuntimeException>
|
||||||
|
long checkIndex(long index, long length,
|
||||||
|
BiFunction<String, List<Number>, X> oobef) {
|
||||||
|
if (index < 0 || index >= length)
|
||||||
|
throw outOfBoundsCheckIndex(oobef, index, length);
|
||||||
|
return index;
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Checks if the sub-range from {@code fromIndex} (inclusive) to
|
||||||
|
* {@code toIndex} (exclusive) is within the bounds of range from {@code 0}
|
||||||
|
* (inclusive) to {@code length} (exclusive).
|
||||||
|
*
|
||||||
|
* <p>The sub-range is defined to be out of bounds if any of the following
|
||||||
|
* inequalities is true:
|
||||||
|
* <ul>
|
||||||
|
* <li>{@code fromIndex < 0}</li>
|
||||||
|
* <li>{@code fromIndex > toIndex}</li>
|
||||||
|
* <li>{@code toIndex > length}</li>
|
||||||
|
* <li>{@code length < 0}, which is implied from the former inequalities</li>
|
||||||
|
* </ul>
|
||||||
|
*
|
||||||
|
* <p>If the sub-range is out of bounds, then a runtime exception is
|
||||||
|
* thrown that is the result of applying the following arguments to the
|
||||||
|
* exception formatter: the name of this method, {@code checkFromToIndex};
|
||||||
|
* and an unmodifiable list of longs whose values are, in order, the
|
||||||
|
* out-of-bounds arguments {@code fromIndex}, {@code toIndex}, and {@code length}.
|
||||||
|
*
|
||||||
|
* @param <X> the type of runtime exception to throw if the arguments are
|
||||||
|
* out of bounds
|
||||||
|
* @param fromIndex the lower-bound (inclusive) of the sub-range
|
||||||
|
* @param toIndex the upper-bound (exclusive) of the sub-range
|
||||||
|
* @param length the upper-bound (exclusive) the range
|
||||||
|
* @param oobef the exception formatter that when applied with this
|
||||||
|
* method name and out-of-bounds arguments returns a runtime
|
||||||
|
* exception. If {@code null} or returns {@code null} then, it is as
|
||||||
|
* if an exception formatter produced from an invocation of
|
||||||
|
* {@code outOfBoundsExceptionFormatter(IndexOutOfBounds::new)} is used
|
||||||
|
* instead (though it may be more efficient).
|
||||||
|
* Exceptions thrown by the formatter are relayed to the caller.
|
||||||
|
* @return {@code fromIndex} if the sub-range within bounds of the range
|
||||||
|
* @throws X if the sub-range is out of bounds and the exception factory
|
||||||
|
* function is non-{@code null}
|
||||||
|
* @throws IndexOutOfBoundsException if the sub-range is out of bounds and
|
||||||
|
* the exception factory function is {@code null}
|
||||||
|
* @since 16
|
||||||
|
*/
|
||||||
|
public static <X extends RuntimeException>
|
||||||
|
long checkFromToIndex(long fromIndex, long toIndex, long length,
|
||||||
|
BiFunction<String, List<Number>, X> oobef) {
|
||||||
|
if (fromIndex < 0 || fromIndex > toIndex || toIndex > length)
|
||||||
|
throw outOfBoundsCheckFromToIndex(oobef, fromIndex, toIndex, length);
|
||||||
|
return fromIndex;
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Checks if the sub-range from {@code fromIndex} (inclusive) to
|
||||||
|
* {@code fromIndex + size} (exclusive) is within the bounds of range from
|
||||||
|
* {@code 0} (inclusive) to {@code length} (exclusive).
|
||||||
|
*
|
||||||
|
* <p>The sub-range is defined to be out of bounds if any of the following
|
||||||
|
* inequalities is true:
|
||||||
|
* <ul>
|
||||||
|
* <li>{@code fromIndex < 0}</li>
|
||||||
|
* <li>{@code size < 0}</li>
|
||||||
|
* <li>{@code fromIndex + size > length}, taking into account integer overflow</li>
|
||||||
|
* <li>{@code length < 0}, which is implied from the former inequalities</li>
|
||||||
|
* </ul>
|
||||||
|
*
|
||||||
|
* <p>If the sub-range is out of bounds, then a runtime exception is
|
||||||
|
* thrown that is the result of applying the following arguments to the
|
||||||
|
* exception formatter: the name of this method, {@code checkFromIndexSize};
|
||||||
|
* and an unmodifiable list of longs whose values are, in order, the
|
||||||
|
* out-of-bounds arguments {@code fromIndex}, {@code size}, and
|
||||||
|
* {@code length}.
|
||||||
|
*
|
||||||
|
* @param <X> the type of runtime exception to throw if the arguments are
|
||||||
|
* out of bounds
|
||||||
|
* @param fromIndex the lower-bound (inclusive) of the sub-interval
|
||||||
|
* @param size the size of the sub-range
|
||||||
|
* @param length the upper-bound (exclusive) of the range
|
||||||
|
* @param oobef the exception formatter that when applied with this
|
||||||
|
* method name and out-of-bounds arguments returns a runtime
|
||||||
|
* exception. If {@code null} or returns {@code null} then, it is as
|
||||||
|
* if an exception formatter produced from an invocation of
|
||||||
|
* {@code outOfBoundsExceptionFormatter(IndexOutOfBounds::new)} is used
|
||||||
|
* instead (though it may be more efficient).
|
||||||
|
* Exceptions thrown by the formatter are relayed to the caller.
|
||||||
|
* @return {@code fromIndex} if the sub-range within bounds of the range
|
||||||
|
* @throws X if the sub-range is out of bounds and the exception factory
|
||||||
|
* function is non-{@code null}
|
||||||
|
* @throws IndexOutOfBoundsException if the sub-range is out of bounds and
|
||||||
|
* the exception factory function is {@code null}
|
||||||
|
* @since 16
|
||||||
|
*/
|
||||||
|
public static <X extends RuntimeException>
|
||||||
|
long checkFromIndexSize(long fromIndex, long size, long length,
|
||||||
|
BiFunction<String, List<Number>, X> oobef) {
|
||||||
if ((length | fromIndex | size) < 0 || size > length - fromIndex)
|
if ((length | fromIndex | size) < 0 || size > length - fromIndex)
|
||||||
throw outOfBoundsCheckFromIndexSize(oobef, fromIndex, size, length);
|
throw outOfBoundsCheckFromIndexSize(oobef, fromIndex, size, length);
|
||||||
return fromIndex;
|
return fromIndex;
|
||||||
|
@ -1,5 +1,5 @@
|
|||||||
/*
|
/*
|
||||||
* Copyright (c) 2018, Oracle and/or its affiliates. All rights reserved.
|
* Copyright (c) 2018, 2020, Oracle and/or its affiliates. All rights reserved.
|
||||||
* DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER.
|
* DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER.
|
||||||
*
|
*
|
||||||
* This code is free software; you can redistribute it and/or modify it
|
* This code is free software; you can redistribute it and/or modify it
|
||||||
@ -37,7 +37,7 @@ import jdk.internal.util.Preconditions;
|
|||||||
|
|
||||||
public final class ArrayUtil {
|
public final class ArrayUtil {
|
||||||
|
|
||||||
private static final BiFunction<String, List<Integer>,
|
private static final BiFunction<String, List<Number>,
|
||||||
ArrayIndexOutOfBoundsException> AIOOBE_SUPPLIER =
|
ArrayIndexOutOfBoundsException> AIOOBE_SUPPLIER =
|
||||||
Preconditions.outOfBoundsExceptionFormatter
|
Preconditions.outOfBoundsExceptionFormatter
|
||||||
(ArrayIndexOutOfBoundsException::new);
|
(ArrayIndexOutOfBoundsException::new);
|
||||||
|
@ -436,6 +436,7 @@ public class CheckGraalIntrinsics extends GraalTest {
|
|||||||
"java/lang/Math.copySign(FF)F",
|
"java/lang/Math.copySign(FF)F",
|
||||||
"java/lang/Math.signum(D)D",
|
"java/lang/Math.signum(D)D",
|
||||||
"java/lang/Math.signum(F)F",
|
"java/lang/Math.signum(F)F",
|
||||||
|
"jdk/internal/util/Preconditions.checkIndex(JJLjava/util/function/BiFunction;)J",
|
||||||
"sun/security/provider/MD5.implCompress0([BI)V");
|
"sun/security/provider/MD5.implCompress0([BI)V");
|
||||||
if (config.useBase64Intrinsics()) {
|
if (config.useBase64Intrinsics()) {
|
||||||
// Currently implemented on ppc64le only, but could be implemented on others
|
// Currently implemented on ppc64le only, but could be implemented on others
|
||||||
|
133
test/hotspot/jtreg/compiler/intrinsics/TestCheckIndex.java
Normal file
133
test/hotspot/jtreg/compiler/intrinsics/TestCheckIndex.java
Normal file
@ -0,0 +1,133 @@
|
|||||||
|
/*
|
||||||
|
* Copyright (c) 2020, Red Hat, 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
|
||||||
|
* @bug 8255150
|
||||||
|
* @summary Add utility methods to check long indexes and ranges
|
||||||
|
* @requires vm.compiler2.enabled
|
||||||
|
* @requires vm.compMode != "Xcomp"
|
||||||
|
* @library /test/lib /
|
||||||
|
* @build sun.hotspot.WhiteBox
|
||||||
|
* @run driver ClassFileInstaller sun.hotspot.WhiteBox
|
||||||
|
*
|
||||||
|
* @run main/othervm -ea -Xbootclasspath/a:. -XX:+UnlockDiagnosticVMOptions -XX:+WhiteBoxAPI -XX:-BackgroundCompilation TestCheckIndex
|
||||||
|
*
|
||||||
|
*/
|
||||||
|
|
||||||
|
import java.util.Objects;
|
||||||
|
import sun.hotspot.WhiteBox;
|
||||||
|
import java.lang.reflect.Method;
|
||||||
|
import compiler.whitebox.CompilerWhiteBoxTest;
|
||||||
|
|
||||||
|
public class TestCheckIndex {
|
||||||
|
private static final WhiteBox WHITE_BOX = WhiteBox.getWhiteBox();
|
||||||
|
|
||||||
|
public static void main(String[] args) throws NoSuchMethodException {
|
||||||
|
Objects.checkIndex(0, 10); // Load class
|
||||||
|
Method m1 = TestCheckIndex.class.getDeclaredMethod("test1", int.class, int.class);
|
||||||
|
Method m2 = TestCheckIndex.class.getDeclaredMethod("test2", long.class, long.class);
|
||||||
|
Method m3 = TestCheckIndex.class.getDeclaredMethod("test3", int.class, int.class);
|
||||||
|
Method m4 = TestCheckIndex.class.getDeclaredMethod("test4", long.class, long.class);
|
||||||
|
assert m1 != null && m2 != null && m3 != null && m4 != null;
|
||||||
|
WHITE_BOX.enqueueMethodForCompilation(m1, CompilerWhiteBoxTest.COMP_LEVEL_FULL_OPTIMIZATION);
|
||||||
|
if (!WHITE_BOX.isMethodCompiled(m1)) {
|
||||||
|
throw new RuntimeException("should be compiled");
|
||||||
|
}
|
||||||
|
WHITE_BOX.enqueueMethodForCompilation(m2, CompilerWhiteBoxTest.COMP_LEVEL_FULL_OPTIMIZATION);
|
||||||
|
if (!WHITE_BOX.isMethodCompiled(m2)) {
|
||||||
|
throw new RuntimeException("should be compiled");
|
||||||
|
}
|
||||||
|
WHITE_BOX.enqueueMethodForCompilation(m3, CompilerWhiteBoxTest.COMP_LEVEL_FULL_OPTIMIZATION);
|
||||||
|
if (!WHITE_BOX.isMethodCompiled(m3)) {
|
||||||
|
throw new RuntimeException("should be compiled");
|
||||||
|
}
|
||||||
|
WHITE_BOX.enqueueMethodForCompilation(m4, CompilerWhiteBoxTest.COMP_LEVEL_FULL_OPTIMIZATION);
|
||||||
|
if (!WHITE_BOX.isMethodCompiled(m4)) {
|
||||||
|
throw new RuntimeException("should be compiled");
|
||||||
|
}
|
||||||
|
|
||||||
|
if (test1(0, 10) != 0) {
|
||||||
|
throw new RuntimeException("incorrect result");
|
||||||
|
}
|
||||||
|
if (!WHITE_BOX.isMethodCompiled(m1)) {
|
||||||
|
throw new RuntimeException("should still be compiled");
|
||||||
|
}
|
||||||
|
if (test2(0, 10) != 0) {
|
||||||
|
throw new RuntimeException("incorrect result");
|
||||||
|
}
|
||||||
|
if (!WHITE_BOX.isMethodCompiled(m2)) {
|
||||||
|
throw new RuntimeException("should still be compiled");
|
||||||
|
}
|
||||||
|
|
||||||
|
try {
|
||||||
|
test1(0, -10);
|
||||||
|
throw new RuntimeException("exception not thrown");
|
||||||
|
} catch (IndexOutOfBoundsException ioobe) {
|
||||||
|
}
|
||||||
|
if (WHITE_BOX.isMethodCompiled(m1)) {
|
||||||
|
throw new RuntimeException("should have deoptimized");
|
||||||
|
}
|
||||||
|
try {
|
||||||
|
test2(0, -10);
|
||||||
|
throw new RuntimeException("exception not thrown");
|
||||||
|
} catch (IndexOutOfBoundsException ioobe) {
|
||||||
|
}
|
||||||
|
if (WHITE_BOX.isMethodCompiled(m2)) {
|
||||||
|
throw new RuntimeException("should have deoptimized");
|
||||||
|
}
|
||||||
|
|
||||||
|
try {
|
||||||
|
test3(42, 10);
|
||||||
|
throw new RuntimeException("exception not thrown");
|
||||||
|
} catch (IndexOutOfBoundsException ioobe) {
|
||||||
|
}
|
||||||
|
if (WHITE_BOX.isMethodCompiled(m3)) {
|
||||||
|
throw new RuntimeException("should have deoptimized");
|
||||||
|
}
|
||||||
|
try {
|
||||||
|
test4(42, 10);
|
||||||
|
throw new RuntimeException("exception not thrown");
|
||||||
|
} catch (IndexOutOfBoundsException ioobe) {
|
||||||
|
}
|
||||||
|
if (WHITE_BOX.isMethodCompiled(m4)) {
|
||||||
|
throw new RuntimeException("should have deoptimized");
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
static int test1(int index, int length) {
|
||||||
|
return Objects.checkIndex(index, length);
|
||||||
|
}
|
||||||
|
|
||||||
|
static long test2(long index, long length) {
|
||||||
|
return Objects.checkIndex(index, length);
|
||||||
|
}
|
||||||
|
|
||||||
|
static int test3(int index, int length) {
|
||||||
|
return Objects.checkIndex(index, length);
|
||||||
|
}
|
||||||
|
|
||||||
|
static long test4(long index, long length) {
|
||||||
|
return Objects.checkIndex(index, length);
|
||||||
|
}
|
||||||
|
}
|
@ -1,5 +1,5 @@
|
|||||||
/*
|
/*
|
||||||
* Copyright (c) 2015, 2016, Oracle and/or its affiliates. All rights reserved.
|
* Copyright (c) 2015, 2020, Oracle and/or its affiliates. All rights reserved.
|
||||||
* DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER.
|
* DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER.
|
||||||
*
|
*
|
||||||
* This code is free software; you can redistribute it and/or modify it
|
* This code is free software; you can redistribute it and/or modify it
|
||||||
@ -23,7 +23,7 @@
|
|||||||
|
|
||||||
/**
|
/**
|
||||||
* @test
|
* @test
|
||||||
* @summary Objects.checkIndex/jdk.internal.util.Preconditions.checkIndex tests
|
* @summary Objects.checkIndex/jdk.internal.util.Preconditions.checkIndex tests for int values
|
||||||
* @run testng CheckIndex
|
* @run testng CheckIndex
|
||||||
* @bug 8135248 8142493 8155794
|
* @bug 8135248 8142493 8155794
|
||||||
* @modules java.base/jdk.internal.util
|
* @modules java.base/jdk.internal.util
|
||||||
@ -50,7 +50,7 @@ public class CheckIndex {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
static BiFunction<String, List<Integer>, AssertingOutOfBoundsException> assertingOutOfBounds(
|
static BiFunction<String, List<Number>, AssertingOutOfBoundsException> assertingOutOfBounds(
|
||||||
String message, String expCheckKind, Integer... expArgs) {
|
String message, String expCheckKind, Integer... expArgs) {
|
||||||
return (checkKind, args) -> {
|
return (checkKind, args) -> {
|
||||||
assertEquals(checkKind, expCheckKind);
|
assertEquals(checkKind, expCheckKind);
|
||||||
@ -64,7 +64,7 @@ public class CheckIndex {
|
|||||||
};
|
};
|
||||||
}
|
}
|
||||||
|
|
||||||
static BiFunction<String, List<Integer>, AssertingOutOfBoundsException> assertingOutOfBoundsReturnNull(
|
static BiFunction<String, List<Number>, AssertingOutOfBoundsException> assertingOutOfBoundsReturnNull(
|
||||||
String expCheckKind, Integer... expArgs) {
|
String expCheckKind, Integer... expArgs) {
|
||||||
return (checkKind, args) -> {
|
return (checkKind, args) -> {
|
||||||
assertEquals(checkKind, expCheckKind);
|
assertEquals(checkKind, expCheckKind);
|
||||||
@ -86,11 +86,7 @@ public class CheckIndex {
|
|||||||
l.add(new Object[]{index, length, withinBounds});
|
l.add(new Object[]{index, length, withinBounds});
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
return l.toArray(new Object[0][0]);
|
return l.toArray(Object[][]::new);
|
||||||
}
|
|
||||||
|
|
||||||
interface X {
|
|
||||||
int apply(int a, int b, int c);
|
|
||||||
}
|
}
|
||||||
|
|
||||||
@Test(dataProvider = "checkIndexProvider")
|
@Test(dataProvider = "checkIndexProvider")
|
||||||
@ -152,7 +148,7 @@ public class CheckIndex {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
return l.toArray(new Object[0][0]);
|
return l.toArray(Object[][]::new);
|
||||||
}
|
}
|
||||||
|
|
||||||
@Test(dataProvider = "checkFromToIndexProvider")
|
@Test(dataProvider = "checkFromToIndexProvider")
|
||||||
@ -221,7 +217,7 @@ public class CheckIndex {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
return l.toArray(new Object[0][0]);
|
return l.toArray(Object[][]::new);
|
||||||
}
|
}
|
||||||
|
|
||||||
@Test(dataProvider = "checkFromIndexSizeProvider")
|
@Test(dataProvider = "checkFromIndexSizeProvider")
|
||||||
@ -269,7 +265,7 @@ public class CheckIndex {
|
|||||||
|
|
||||||
@Test
|
@Test
|
||||||
public void uniqueMessagesForCheckKinds() {
|
public void uniqueMessagesForCheckKinds() {
|
||||||
BiFunction<String, List<Integer>, IndexOutOfBoundsException> f =
|
BiFunction<String, List<Number>, IndexOutOfBoundsException> f =
|
||||||
Preconditions.outOfBoundsExceptionFormatter(IndexOutOfBoundsException::new);
|
Preconditions.outOfBoundsExceptionFormatter(IndexOutOfBoundsException::new);
|
||||||
|
|
||||||
List<String> messages = new ArrayList<>();
|
List<String> messages = new ArrayList<>();
|
||||||
|
287
test/jdk/java/util/Objects/CheckLongIndex.java
Normal file
287
test/jdk/java/util/Objects/CheckLongIndex.java
Normal file
@ -0,0 +1,287 @@
|
|||||||
|
/*
|
||||||
|
* Copyright (c) 2020, 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
|
||||||
|
* 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 Objects.checkIndex/jdk.internal.util.Preconditions.checkIndex tests for long values
|
||||||
|
* @run testng CheckLongIndex
|
||||||
|
* @modules java.base/jdk.internal.util
|
||||||
|
*/
|
||||||
|
|
||||||
|
import jdk.internal.util.Preconditions;
|
||||||
|
import org.testng.annotations.DataProvider;
|
||||||
|
import org.testng.annotations.Test;
|
||||||
|
|
||||||
|
import java.util.ArrayList;
|
||||||
|
import java.util.List;
|
||||||
|
import java.util.Objects;
|
||||||
|
import java.util.function.BiConsumer;
|
||||||
|
import java.util.function.BiFunction;
|
||||||
|
import java.util.function.LongSupplier;
|
||||||
|
|
||||||
|
import static org.testng.Assert.*;
|
||||||
|
|
||||||
|
public class CheckLongIndex {
|
||||||
|
|
||||||
|
static class AssertingOutOfBoundsException extends RuntimeException {
|
||||||
|
public AssertingOutOfBoundsException(String message) {
|
||||||
|
super(message);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
static BiFunction<String, List<Number>, AssertingOutOfBoundsException> assertingOutOfBounds(
|
||||||
|
String message, String expCheckKind, Long... expArgs) {
|
||||||
|
return (checkKind, args) -> {
|
||||||
|
assertEquals(checkKind, expCheckKind);
|
||||||
|
assertEquals(args, List.of(expArgs));
|
||||||
|
try {
|
||||||
|
args.clear();
|
||||||
|
fail("Out of bounds List<Long> argument should be unmodifiable");
|
||||||
|
} catch (Exception e) {
|
||||||
|
}
|
||||||
|
return new AssertingOutOfBoundsException(message);
|
||||||
|
};
|
||||||
|
}
|
||||||
|
|
||||||
|
static BiFunction<String, List<Number>, AssertingOutOfBoundsException> assertingOutOfBoundsReturnNull(
|
||||||
|
String expCheckKind, Long... expArgs) {
|
||||||
|
return (checkKind, args) -> {
|
||||||
|
assertEquals(checkKind, expCheckKind);
|
||||||
|
assertEquals(args, List.of(expArgs));
|
||||||
|
return null;
|
||||||
|
};
|
||||||
|
}
|
||||||
|
|
||||||
|
static final long[] VALUES = {0, 1, Long.MAX_VALUE - 1, Long.MAX_VALUE, -1, Long.MIN_VALUE + 1, Long.MIN_VALUE};
|
||||||
|
|
||||||
|
@DataProvider
|
||||||
|
static Object[][] checkIndexProvider() {
|
||||||
|
List<Object[]> l = new ArrayList<>();
|
||||||
|
for (long index : VALUES) {
|
||||||
|
for (long length : VALUES) {
|
||||||
|
boolean withinBounds = index >= 0 &&
|
||||||
|
length >= 0 &&
|
||||||
|
index < length;
|
||||||
|
l.add(new Object[]{index, length, withinBounds});
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return l.toArray(Object[][]::new);
|
||||||
|
}
|
||||||
|
|
||||||
|
@Test(dataProvider = "checkIndexProvider")
|
||||||
|
public void testCheckIndex(long index, long length, boolean withinBounds) {
|
||||||
|
String expectedMessage = withinBounds
|
||||||
|
? null
|
||||||
|
: Preconditions.outOfBoundsExceptionFormatter(IndexOutOfBoundsException::new).
|
||||||
|
apply("checkIndex", List.of(index, length)).getMessage();
|
||||||
|
|
||||||
|
BiConsumer<Class<? extends RuntimeException>, LongSupplier> checker = (ec, s) -> {
|
||||||
|
try {
|
||||||
|
long rIndex = s.getAsLong();
|
||||||
|
if (!withinBounds)
|
||||||
|
fail(String.format(
|
||||||
|
"Index %d is out of bounds of [0, %d), but was reported to be within bounds", index, length));
|
||||||
|
assertEquals(rIndex, index);
|
||||||
|
}
|
||||||
|
catch (RuntimeException e) {
|
||||||
|
assertTrue(ec.isInstance(e));
|
||||||
|
if (withinBounds)
|
||||||
|
fail(String.format(
|
||||||
|
"Index %d is within bounds of [0, %d), but was reported to be out of bounds", index, length));
|
||||||
|
else
|
||||||
|
assertEquals(e.getMessage(), expectedMessage);
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
|
checker.accept(AssertingOutOfBoundsException.class,
|
||||||
|
() -> Preconditions.checkIndex(index, length,
|
||||||
|
assertingOutOfBounds(expectedMessage, "checkIndex", index, length)));
|
||||||
|
checker.accept(IndexOutOfBoundsException.class,
|
||||||
|
() -> Preconditions.checkIndex(index, length,
|
||||||
|
assertingOutOfBoundsReturnNull("checkIndex", index, length)));
|
||||||
|
checker.accept(IndexOutOfBoundsException.class,
|
||||||
|
() -> Preconditions.checkIndex(index, length, null));
|
||||||
|
checker.accept(IndexOutOfBoundsException.class,
|
||||||
|
() -> Objects.checkIndex(index, length));
|
||||||
|
checker.accept(ArrayIndexOutOfBoundsException.class,
|
||||||
|
() -> Preconditions.checkIndex(index, length,
|
||||||
|
Preconditions.outOfBoundsExceptionFormatter(ArrayIndexOutOfBoundsException::new)));
|
||||||
|
checker.accept(StringIndexOutOfBoundsException.class,
|
||||||
|
() -> Preconditions.checkIndex(index, length,
|
||||||
|
Preconditions.outOfBoundsExceptionFormatter(StringIndexOutOfBoundsException::new)));
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
@DataProvider
|
||||||
|
static Object[][] checkFromToIndexProvider() {
|
||||||
|
List<Object[]> l = new ArrayList<>();
|
||||||
|
for (long fromIndex : VALUES) {
|
||||||
|
for (long toIndex : VALUES) {
|
||||||
|
for (long length : VALUES) {
|
||||||
|
boolean withinBounds = fromIndex >= 0 &&
|
||||||
|
toIndex >= 0 &&
|
||||||
|
length >= 0 &&
|
||||||
|
fromIndex <= toIndex &&
|
||||||
|
toIndex <= length;
|
||||||
|
l.add(new Object[]{fromIndex, toIndex, length, withinBounds});
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return l.toArray(Object[][]::new);
|
||||||
|
}
|
||||||
|
|
||||||
|
@Test(dataProvider = "checkFromToIndexProvider")
|
||||||
|
public void testCheckFromToIndex(long fromIndex, long toIndex, long length, boolean withinBounds) {
|
||||||
|
String expectedMessage = withinBounds
|
||||||
|
? null
|
||||||
|
: Preconditions.outOfBoundsExceptionFormatter(IndexOutOfBoundsException::new).
|
||||||
|
apply("checkFromToIndex", List.of(fromIndex, toIndex, length)).getMessage();
|
||||||
|
|
||||||
|
BiConsumer<Class<? extends RuntimeException>, LongSupplier> check = (ec, s) -> {
|
||||||
|
try {
|
||||||
|
long rIndex = s.getAsLong();
|
||||||
|
if (!withinBounds)
|
||||||
|
fail(String.format(
|
||||||
|
"Range [%d, %d) is out of bounds of [0, %d), but was reported to be withing bounds", fromIndex, toIndex, length));
|
||||||
|
assertEquals(rIndex, fromIndex);
|
||||||
|
}
|
||||||
|
catch (RuntimeException e) {
|
||||||
|
assertTrue(ec.isInstance(e));
|
||||||
|
if (withinBounds)
|
||||||
|
fail(String.format(
|
||||||
|
"Range [%d, %d) is within bounds of [0, %d), but was reported to be out of bounds", fromIndex, toIndex, length));
|
||||||
|
else
|
||||||
|
assertEquals(e.getMessage(), expectedMessage);
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
|
check.accept(AssertingOutOfBoundsException.class,
|
||||||
|
() -> Preconditions.checkFromToIndex(fromIndex, toIndex, length,
|
||||||
|
assertingOutOfBounds(expectedMessage, "checkFromToIndex", fromIndex, toIndex, length)));
|
||||||
|
check.accept(IndexOutOfBoundsException.class,
|
||||||
|
() -> Preconditions.checkFromToIndex(fromIndex, toIndex, length,
|
||||||
|
assertingOutOfBoundsReturnNull("checkFromToIndex", fromIndex, toIndex, length)));
|
||||||
|
check.accept(IndexOutOfBoundsException.class,
|
||||||
|
() -> Preconditions.checkFromToIndex(fromIndex, toIndex, length, null));
|
||||||
|
check.accept(IndexOutOfBoundsException.class,
|
||||||
|
() -> Objects.checkFromToIndex(fromIndex, toIndex, length));
|
||||||
|
check.accept(ArrayIndexOutOfBoundsException.class,
|
||||||
|
() -> Preconditions.checkFromToIndex(fromIndex, toIndex, length,
|
||||||
|
Preconditions.outOfBoundsExceptionFormatter(ArrayIndexOutOfBoundsException::new)));
|
||||||
|
check.accept(StringIndexOutOfBoundsException.class,
|
||||||
|
() -> Preconditions.checkFromToIndex(fromIndex, toIndex, length,
|
||||||
|
Preconditions.outOfBoundsExceptionFormatter(StringIndexOutOfBoundsException::new)));
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
@DataProvider
|
||||||
|
static Object[][] checkFromIndexSizeProvider() {
|
||||||
|
List<Object[]> l = new ArrayList<>();
|
||||||
|
for (long fromIndex : VALUES) {
|
||||||
|
for (long size : VALUES) {
|
||||||
|
for (long length : VALUES) {
|
||||||
|
long toIndex = fromIndex + size;
|
||||||
|
|
||||||
|
boolean withinBounds = fromIndex >= 0L &&
|
||||||
|
size >= 0L &&
|
||||||
|
length >= 0L &&
|
||||||
|
fromIndex <= toIndex && // overflow
|
||||||
|
toIndex <= length;
|
||||||
|
l.add(new Object[]{fromIndex, size, length, withinBounds});
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return l.toArray(Object[][]::new);
|
||||||
|
}
|
||||||
|
|
||||||
|
@Test(dataProvider = "checkFromIndexSizeProvider")
|
||||||
|
public void testCheckFromIndexSize(long fromIndex, long size, long length, boolean withinBounds) {
|
||||||
|
String expectedMessage = withinBounds
|
||||||
|
? null
|
||||||
|
: Preconditions.outOfBoundsExceptionFormatter(IndexOutOfBoundsException::new).
|
||||||
|
apply("checkFromIndexSize", List.of(fromIndex, size, length)).getMessage();
|
||||||
|
|
||||||
|
BiConsumer<Class<? extends RuntimeException>, LongSupplier> check = (ec, s) -> {
|
||||||
|
try {
|
||||||
|
long rIndex = s.getAsLong();
|
||||||
|
if (!withinBounds)
|
||||||
|
fail(String.format(
|
||||||
|
"Range [%d, %d + %d) is out of bounds of [0, %d), but was reported to be withing bounds", fromIndex, fromIndex, size, length));
|
||||||
|
assertEquals(rIndex, fromIndex);
|
||||||
|
}
|
||||||
|
catch (RuntimeException e) {
|
||||||
|
assertTrue(ec.isInstance(e));
|
||||||
|
if (withinBounds)
|
||||||
|
fail(String.format(
|
||||||
|
"Range [%d, %d + %d) is within bounds of [0, %d), but was reported to be out of bounds", fromIndex, fromIndex, size, length));
|
||||||
|
else
|
||||||
|
assertEquals(e.getMessage(), expectedMessage);
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
|
check.accept(AssertingOutOfBoundsException.class,
|
||||||
|
() -> Preconditions.checkFromIndexSize(fromIndex, size, length,
|
||||||
|
assertingOutOfBounds(expectedMessage, "checkFromIndexSize", fromIndex, size, length)));
|
||||||
|
check.accept(IndexOutOfBoundsException.class,
|
||||||
|
() -> Preconditions.checkFromIndexSize(fromIndex, size, length,
|
||||||
|
assertingOutOfBoundsReturnNull("checkFromIndexSize", fromIndex, size, length)));
|
||||||
|
check.accept(IndexOutOfBoundsException.class,
|
||||||
|
() -> Preconditions.checkFromIndexSize(fromIndex, size, length, null));
|
||||||
|
check.accept(IndexOutOfBoundsException.class,
|
||||||
|
() -> Objects.checkFromIndexSize(fromIndex, size, length));
|
||||||
|
check.accept(ArrayIndexOutOfBoundsException.class,
|
||||||
|
() -> Preconditions.checkFromIndexSize(fromIndex, size, length,
|
||||||
|
Preconditions.outOfBoundsExceptionFormatter(ArrayIndexOutOfBoundsException::new)));
|
||||||
|
check.accept(StringIndexOutOfBoundsException.class,
|
||||||
|
() -> Preconditions.checkFromIndexSize(fromIndex, size, length,
|
||||||
|
Preconditions.outOfBoundsExceptionFormatter(StringIndexOutOfBoundsException::new)));
|
||||||
|
}
|
||||||
|
|
||||||
|
@Test
|
||||||
|
public void uniqueMessagesForCheckKinds() {
|
||||||
|
BiFunction<String, List<Number>, IndexOutOfBoundsException> f =
|
||||||
|
Preconditions.outOfBoundsExceptionFormatter(IndexOutOfBoundsException::new);
|
||||||
|
|
||||||
|
List<String> messages = new ArrayList<>();
|
||||||
|
// Exact arguments
|
||||||
|
messages.add(f.apply("checkIndex", List.of(-1L, 0L)).getMessage());
|
||||||
|
messages.add(f.apply("checkFromToIndex", List.of(-1L, 0L, 0L)).getMessage());
|
||||||
|
messages.add(f.apply("checkFromIndexSize", List.of(-1L, 0L, 0L)).getMessage());
|
||||||
|
// Unknown check kind
|
||||||
|
messages.add(f.apply("checkUnknown", List.of(-1L, 0L, 0L)).getMessage());
|
||||||
|
// Known check kind with more arguments
|
||||||
|
messages.add(f.apply("checkIndex", List.of(-1L, 0L, 0L)).getMessage());
|
||||||
|
messages.add(f.apply("checkFromToIndex", List.of(-1L, 0L, 0L, 0L)).getMessage());
|
||||||
|
messages.add(f.apply("checkFromIndexSize", List.of(-1L, 0L, 0L, 0L)).getMessage());
|
||||||
|
// Known check kind with fewer arguments
|
||||||
|
messages.add(f.apply("checkIndex", List.of(-1L)).getMessage());
|
||||||
|
messages.add(f.apply("checkFromToIndex", List.of(-1L, 0L)).getMessage());
|
||||||
|
messages.add(f.apply("checkFromIndexSize", List.of(-1L, 0L)).getMessage());
|
||||||
|
// Null arguments
|
||||||
|
messages.add(f.apply(null, null).getMessage());
|
||||||
|
messages.add(f.apply("checkNullArguments", null).getMessage());
|
||||||
|
messages.add(f.apply(null, List.of(-1L)).getMessage());
|
||||||
|
|
||||||
|
assertEquals(messages.size(), messages.stream().distinct().count());
|
||||||
|
}
|
||||||
|
}
|
Loading…
Reference in New Issue
Block a user