-
Notifications
You must be signed in to change notification settings - Fork 109
New issue
Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.
By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.
Already on GitHub? Sign in to your account
8304310: Initial compilers and runtime handling for multifield backed vectors. #833
Changes from 1 commit
b9627b4
cdbffd6
6266113
6a213f9
01cc2b6
245e6b6
b5dd30a
6e73591
5902b90
23fa574
638959b
e79c0a7
a5d0e15
21bea77
4bf661e
c68a767
File filter
Filter by extension
Conversations
Jump to
Diff view
Diff view
There are no files selected for viewing
Original file line number | Diff line number | Diff line change |
---|---|---|
|
@@ -921,7 +921,7 @@ void Compile::return_values(JVMState* jvms) { | |
kit.inc_sp(-ret_size); // pop the return value(s) | ||
kit.sync_jvms(); | ||
Node* res = kit.argument(0); | ||
if (res->isa_InlineType() && VectorSupport::skip_value_scalarization(res->as_InlineType()->inline_klass())) { | ||
if (res->isa_InlineType() && VectorSupport::skip_value_scalarization(res->as_InlineType()->inline_klass()->get_InlineKlass())) { | ||
There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. I think in general multi-field fields should be scalarised differently from normal fields, since they represent an array and ideally should be packed together on the stack instead of passing separately. As a result, for now it may be better to reject scalarisation of types with multi-fields altogether. There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. Agree. We will eventually need to extend new return calling convention to support vectors. |
||
InlineTypeNode* vt = res->as_InlineType(); | ||
assert(vt->is_buffered(), ""); | ||
ret->add_req(vt->get_oop()); | ||
|
@@ -2357,7 +2357,7 @@ void Parse::return_current(Node* value) { | |
if (!value->is_InlineType()) { | ||
value = InlineTypeNode::make_from_oop(this, value, return_type->inline_klass(), method()->signature()->returns_null_free_inline_type()); | ||
} | ||
if (VectorSupport::skip_value_scalarization(value->as_InlineType()->inline_klass())) { | ||
if (VectorSupport::skip_value_scalarization(value->as_InlineType()->inline_klass()->get_InlineKlass())) { | ||
// Buffer the vector return types, for regular inline object caller expects | ||
// scalarized fields to be passed back. | ||
PreserveReexecuteState preexecs(this); | ||
|
Original file line number | Diff line number | Diff line change |
---|---|---|
|
@@ -34,15 +34,19 @@ | |
#include "prims/vectorSupport.hpp" | ||
|
||
static bool is_vector(ciKlass* klass) { | ||
return VectorSupport::is_vector(klass); | ||
return klass->is_subclass_of(ciEnv::current()->vector_Vector_klass()); | ||
} | ||
|
||
static bool is_vector_payload_mf(ciKlass* klass) { | ||
return klass->is_subclass_of(ciEnv::current()->vector_VectorPayloadMF_klass()); | ||
} | ||
|
||
static bool is_vector_mask(ciKlass* klass) { | ||
return VectorSupport::is_vector_mask(klass); | ||
return klass->is_subclass_of(ciEnv::current()->vector_VectorMask_klass()); | ||
} | ||
|
||
static bool is_vector_shuffle(ciKlass* klass) { | ||
return VectorSupport::is_vector_shuffle(klass); | ||
return klass->is_subclass_of(ciEnv::current()->vector_VectorShuffle_klass()); | ||
} | ||
|
||
void PhaseVector::optimize_vector_boxes() { | ||
|
@@ -316,23 +320,22 @@ void PhaseVector::expand_vbox_node(VectorBoxNode* vec_box) { | |
if (vec_box->outcnt() > 0) { | ||
Node* vbox = vec_box->get_oop(); | ||
Node* vect = vec_box->get_vec(); | ||
Node* result = expand_vbox_node_helper(vec_box, vbox, vect, vec_box->box_type(), vec_box->vec_type()); | ||
Node* result = expand_vbox_node_helper(vbox, vect, vec_box->box_type(), vec_box->vec_type()); | ||
C->gvn_replace_by(vec_box, result); | ||
C->print_method(PHASE_EXPAND_VBOX, 3, vec_box); | ||
} | ||
C->remove_macro_node(vec_box); | ||
} | ||
|
||
Node* PhaseVector::expand_vbox_node_helper(Node* vec_box, | ||
Node* vbox, | ||
Node* PhaseVector::expand_vbox_node_helper(Node* vbox, | ||
Node* vect, | ||
const TypeInstPtr* box_type, | ||
const TypeVect* vect_type) { | ||
if (vbox->is_Phi() && vect->is_Phi()) { | ||
assert(vbox->as_Phi()->region() == vect->as_Phi()->region(), ""); | ||
Node* new_phi = new PhiNode(vbox->as_Phi()->region(), box_type); | ||
for (uint i = 1; i < vbox->req(); i++) { | ||
Node* new_box = expand_vbox_node_helper(vec_box, vbox->in(i), vect->in(i), box_type, vect_type); | ||
Node* new_box = expand_vbox_node_helper(vbox->in(i), vect->in(i), box_type, vect_type); | ||
new_phi->set_req(i, new_box); | ||
} | ||
new_phi = C->initial_gvn()->transform(new_phi); | ||
|
@@ -347,7 +350,7 @@ Node* PhaseVector::expand_vbox_node_helper(Node* vec_box, | |
// move up and are guaranteed to dominate. | ||
Node* new_phi = new PhiNode(vbox->as_Phi()->region(), box_type); | ||
for (uint i = 1; i < vbox->req(); i++) { | ||
Node* new_box = expand_vbox_node_helper(vec_box, vbox->in(i), vect, box_type, vect_type); | ||
Node* new_box = expand_vbox_node_helper(vbox->in(i), vect, box_type, vect_type); | ||
new_phi->set_req(i, new_box); | ||
} | ||
new_phi = C->initial_gvn()->transform(new_phi); | ||
|
@@ -380,12 +383,12 @@ Node* PhaseVector::expand_vbox_alloc_node_vector(VectorBoxAllocateNode* vbox_all | |
Node* klass_node = kit.makecon(klass_type); | ||
Node* buffer_mem = kit.new_instance(klass_node, NULL, NULL, /* deoptimize_on_exception */ true); | ||
|
||
assert(VectorSupport::is_vector(box_klass), ""); | ||
assert(is_vector(box_klass), ""); | ||
ciField* payload_field = box_klass->declared_nonstatic_field_at(0); | ||
int offset = payload_field->offset(); | ||
if (!payload_field->is_flattened()) { | ||
ciInlineKlass* payload_klass = static_cast<ciInlineKlass*>(payload_field->type()); | ||
assert(VectorSupport::is_vector_payload_mf(payload_klass), ""); | ||
assert(is_vector_payload_mf(payload_klass), ""); | ||
ciField* mutifield = payload_klass->declared_nonstatic_field_at(0); | ||
offset += mutifield->offset(); | ||
} | ||
|
@@ -515,41 +518,7 @@ void PhaseVector::expand_vunbox_node_vector(VectorUnboxNode* vec_unbox) { | |
PhaseGVN& gvn = kit.gvn(); | ||
|
||
Node* vec_val_load = get_loaded_payload(vec_unbox); | ||
|
||
There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. Maybe directly merge |
||
if (vec_val_load == NULL) { | ||
Node* obj = vec_unbox->obj(); | ||
const TypeInstPtr* tinst = gvn.type(obj)->isa_instptr(); | ||
assert(VectorSupport::is_vector(tinst->instance_klass()), ""); | ||
ciInlineKlass* box_klass = static_cast<ciInlineKlass*>(tinst->instance_klass()); | ||
|
||
const TypeVect* vt = vec_unbox->bottom_type()->is_vect(); | ||
BasicType bt = vt->element_basic_type(); | ||
int num_elem = vt->length(); | ||
|
||
assert(VectorSupport::is_vector(box_klass), ""); | ||
ciField* payload_field = box_klass->declared_nonstatic_field_at(0); | ||
int offset = payload_field->offset(); | ||
if (!payload_field->is_flattened()) { | ||
ciInlineKlass* payload_klass = static_cast<ciInlineKlass*>(payload_field->type()); | ||
assert(VectorSupport::is_vector_payload_mf(payload_klass), ""); | ||
ciField* mutifield = payload_klass->declared_nonstatic_field_at(0); | ||
offset += mutifield->offset(); | ||
} | ||
|
||
Node* mem = vec_unbox->mem(); | ||
Node* ctrl = vec_unbox->in(0); | ||
Node* vec_adr = gvn.transform(kit.basic_plus_adr(obj, offset)); | ||
const TypePtr *adr_type = gvn.type(vec_adr)->isa_ptr(); | ||
|
||
vec_val_load = LoadVectorNode::make(0, | ||
ctrl, | ||
mem, | ||
vec_adr, | ||
adr_type, | ||
num_elem, | ||
bt); | ||
vec_val_load = gvn.transform(vec_val_load); | ||
} | ||
assert (vec_val_load != NULL, ""); | ||
|
||
C->set_max_vector_size(MAX2(C->max_vector_size(), vec_val_load->bottom_type()->is_vect()->length_in_bytes())); | ||
gvn.hash_delete(vec_unbox); | ||
|
Original file line number | Diff line number | Diff line change |
---|---|---|
|
@@ -29,15 +29,15 @@ | |
#include "opto/phaseX.hpp" | ||
#include "opto/type.hpp" | ||
#include "opto/vectornode.hpp" | ||
#include "ci/ciKlass.hpp" | ||
There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. Style: please reorder this include file. Thanks! |
||
|
||
class PhaseVector : public Phase { | ||
private: | ||
PhaseIterGVN& _igvn; | ||
|
||
void expand_vbox_nodes(); | ||
void expand_vbox_node(VectorBoxNode* vec_box); | ||
Node* expand_vbox_node_helper(Node* vbox, | ||
Node* vbox_alloc, | ||
Node* expand_vbox_node_helper(Node* vbox_alloc, | ||
Node* vect, | ||
const TypeInstPtr* box_type, | ||
const TypeVect* vect_type); | ||
|
Original file line number | Diff line number | Diff line change |
---|---|---|
|
@@ -68,43 +68,22 @@ const char* VectorSupport::svmlname[VectorSupport::NUM_SVML_OP] = { | |
}; | ||
#endif | ||
|
||
|
||
bool VectorSupport::is_vector(ciKlass* klass) { | ||
return klass->is_subclass_of(ciEnv::current()->vector_Vector_klass()); | ||
} | ||
|
||
bool VectorSupport::is_vector(Klass* klass) { | ||
return klass->is_subclass_of(vmClasses::vector_Vector_klass()); | ||
} | ||
|
||
bool VectorSupport::is_vector_payload_mf(ciKlass* klass) { | ||
return klass->is_subclass_of(ciEnv::current()->vector_VectorPayloadMF_klass()); | ||
} | ||
|
||
bool VectorSupport::is_vector_payload_mf(Klass* klass) { | ||
return klass->is_subclass_of(vmClasses::vector_VectorPayloadMF_klass()); | ||
} | ||
|
||
bool VectorSupport::is_vector_mask(ciKlass* klass) { | ||
return klass->is_subclass_of(ciEnv::current()->vector_VectorMask_klass()); | ||
} | ||
|
||
bool VectorSupport::is_vector_mask(Klass* klass) { | ||
return klass->is_subclass_of(vmClasses::vector_VectorMask_klass()); | ||
} | ||
|
||
bool VectorSupport::is_vector_shuffle(ciKlass* klass) { | ||
return klass->is_subclass_of(ciEnv::current()->vector_VectorShuffle_klass()); | ||
} | ||
|
||
bool VectorSupport::is_vector_shuffle(Klass* klass) { | ||
return klass->is_subclass_of(vmClasses::vector_VectorShuffle_klass()); | ||
} | ||
|
||
bool VectorSupport::skip_value_scalarization(ciKlass* klass) { | ||
return VectorSupport::is_vector(klass) || VectorSupport::is_vector_payload_mf(klass); | ||
} | ||
|
||
bool VectorSupport::skip_value_scalarization(Klass* klass) { | ||
return VectorSupport::is_vector(klass) || VectorSupport::is_vector_payload_mf(klass); | ||
} | ||
|
@@ -399,7 +378,6 @@ instanceOop VectorSupport::allocate_vector(InstanceKlass* ik, frame* fr, Registe | |
Handle vbox_h = Handle(THREAD, vbox); | ||
|
||
fieldDescriptor fd; | ||
int elem_size = type2aelembytes(elem_bt); | ||
Symbol* payload_sig = VectorSupport::get_vector_payload_field_signature(elem_bt, num_elem); | ||
There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. Alternatively, you could derive that information from There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more.
find_field need field signature to query the fieldDescriptor. payload is no longer a primitive array but a primitive class instance, we still need pass klass2length() and klass2bt() information to payload signature factory method. There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. Looking up the field by name in the trusted class is also an option. |
||
Klass* def = ik->find_field(vmSymbols::payload_name(), payload_sig, false, &fd); | ||
assert(def != NULL, ""); | ||
|
Original file line number | Diff line number | Diff line change |
---|---|---|
|
@@ -638,7 +638,7 @@ JRT_LEAF(address, SharedRuntime::exception_handler_for_return_address(JavaThread | |
return raw_exception_handler_for_return_address(current, return_address); | ||
JRT_END | ||
|
||
JRT_LEAF(jint, SharedRuntime::is_vector_value_instance(InlineKlass* klass)) | ||
JRT_LEAF(jint, SharedRuntime::skip_value_scalarization(InlineKlass* klass)) | ||
return (jint)VectorSupport::skip_value_scalarization(klass); | ||
There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. The naming is confusing here: There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. DONE |
||
JRT_END | ||
|
||
|
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
I'm wondering if
tf()->returns_inline_type_as_fields()
in followedelse-if
branch has any connection withciInlineKlass::can_be_returned_as_fields()
. IfciInlineKlass::can_be_returned_as_fields()
can decide the result oftf()->returns_inline_type_as_fields()
, then this special handling for bufferredInlineTypeNode
can be removed like before?There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
C2 always scalarizes value objects before returning it, for vector we current do not support scalarization hence I am returning the oop buffer of InlineTypeNode. Without this return value register mask may not comply with value being returned, this will specially be problematic when we return VectorPayloadMF* instances, it only has one field of type vector and return value mask expects a GPR.
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
BTW, Valhalla added a new return calling convention for inline types but it needs to be extended to cover vectors.
https://github.com/openjdk/valhalla/blob/lworld/src/hotspot/cpu/x86/sharedRuntime_x86_64.cpp#L567
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Yeah, make sense to me. What I mean here is whether we can save this new added branch if the
else-if
in line-928 can avoid such multifields scalarization. Not a block, we can refine such code in future if possible.