提交 e9356f59 编写于 作者: S shshahma

8134389: Crash in HotSpot with jvm.dll+0x42b48 ciObjectFactory::create_new_metadata

Summary: Always obtain return type from declared_signature for Invoke::declared_type. TypeCast return value to declared_signature return type for inlined lforms.
Reviewed-by: kvn, kevinw
上级 f0c10dc7
...@@ -1485,6 +1485,21 @@ void GraphBuilder::method_return(Value x) { ...@@ -1485,6 +1485,21 @@ void GraphBuilder::method_return(Value x) {
// Check to see whether we are inlining. If so, Return // Check to see whether we are inlining. If so, Return
// instructions become Gotos to the continuation point. // instructions become Gotos to the continuation point.
if (continuation() != NULL) { if (continuation() != NULL) {
int invoke_bci = state()->caller_state()->bci();
if (x != NULL) {
ciMethod* caller = state()->scope()->caller()->method();
Bytecodes::Code invoke_raw_bc = caller->raw_code_at_bci(invoke_bci);
if (invoke_raw_bc == Bytecodes::_invokehandle || invoke_raw_bc == Bytecodes::_invokedynamic) {
ciType* declared_ret_type = caller->get_declared_signature_at_bci(invoke_bci)->return_type();
if (declared_ret_type->is_klass() && x->exact_type() == NULL &&
x->declared_type() != declared_ret_type && declared_ret_type != compilation()->env()->Object_klass()) {
x = append(new TypeCast(declared_ret_type->as_klass(), x, copy_state_before()));
}
}
}
assert(!method()->is_synchronized() || InlineSynchronizedMethods, "can not inline synchronized methods yet"); assert(!method()->is_synchronized() || InlineSynchronizedMethods, "can not inline synchronized methods yet");
if (compilation()->env()->dtrace_method_probes()) { if (compilation()->env()->dtrace_method_probes()) {
...@@ -1508,7 +1523,6 @@ void GraphBuilder::method_return(Value x) { ...@@ -1508,7 +1523,6 @@ void GraphBuilder::method_return(Value x) {
// State at end of inlined method is the state of the caller // State at end of inlined method is the state of the caller
// without the method parameters on stack, including the // without the method parameters on stack, including the
// return value, if any, of the inlined method on operand stack. // return value, if any, of the inlined method on operand stack.
int invoke_bci = state()->caller_state()->bci();
set_state(state()->caller_state()->copy_for_parsing()); set_state(state()->caller_state()->copy_for_parsing());
if (x != NULL) { if (x != NULL) {
state()->push(x->type(), x); state()->push(x->type(), x);
......
...@@ -360,7 +360,8 @@ void Invoke::state_values_do(ValueVisitor* f) { ...@@ -360,7 +360,8 @@ void Invoke::state_values_do(ValueVisitor* f) {
} }
ciType* Invoke::declared_type() const { ciType* Invoke::declared_type() const {
ciType *t = _target->signature()->return_type(); ciSignature* declared_signature = state()->scope()->method()->get_declared_signature_at_bci(state()->bci());
ciType *t = declared_signature->return_type();
assert(t->basic_type() != T_VOID, "need return value of void method?"); assert(t->basic_type() != T_VOID, "need return value of void method?");
return t; return t;
} }
......
...@@ -243,11 +243,21 @@ class ciMethod : public ciMetadata { ...@@ -243,11 +243,21 @@ class ciMethod : public ciMetadata {
ciField* get_field_at_bci( int bci, bool &will_link); ciField* get_field_at_bci( int bci, bool &will_link);
ciMethod* get_method_at_bci(int bci, bool &will_link, ciSignature* *declared_signature); ciMethod* get_method_at_bci(int bci, bool &will_link, ciSignature* *declared_signature);
ciSignature* get_declared_signature_at_bci(int bci) {
bool ignored_will_link;
ciSignature* declared_signature;
get_method_at_bci(bci, ignored_will_link, &declared_signature);
assert(declared_signature != NULL, "cannot be null");
return declared_signature;
}
ciMethod* get_method_at_bci(int bci) { ciMethod* get_method_at_bci(int bci) {
bool ignored_will_link; bool ignored_will_link;
ciSignature* ignored_declared_signature; ciSignature* ignored_declared_signature;
return get_method_at_bci(bci, ignored_will_link, &ignored_declared_signature); return get_method_at_bci(bci, ignored_will_link, &ignored_declared_signature);
} }
// Given a certain calling environment, find the monomorphic target // Given a certain calling environment, find the monomorphic target
// for the call. Return NULL if the call is not monomorphic in // for the call. Return NULL if the call is not monomorphic in
// its calling environment. // its calling environment.
......
Markdown is supported
0% .
You are about to add 0 people to the discussion. Proceed with caution.
先完成此消息的编辑!
想要评论请 注册