From 9b5e1ddb927989007c1e7ae48039ad3837fdbb9f Mon Sep 17 00:00:00 2001 From: Coleen Phillimore Date: Fri, 29 May 2015 11:35:51 -0400 Subject: [PATCH] 8029567: Clean up linkResolver code Moved non-const reference return values to actual return values, refactored error handling code, remove oop from Method* variable names. Reviewed-by: jiangli, lfoltan, acorn --- hotspot/src/share/vm/ci/ciEnv.cpp | 11 +- hotspot/src/share/vm/ci/ciField.cpp | 8 +- hotspot/src/share/vm/ci/ciMethod.cpp | 14 +- hotspot/src/share/vm/interpreter/bytecode.cpp | 7 +- .../src/share/vm/interpreter/linkResolver.cpp | 890 ++++++++---------- .../src/share/vm/interpreter/linkResolver.hpp | 225 +++-- hotspot/src/share/vm/oops/klassVtable.cpp | 2 +- hotspot/src/share/vm/prims/methodHandles.cpp | 19 +- hotspot/src/share/vm/runtime/javaCalls.cpp | 12 +- hotspot/src/share/vm/runtime/reflection.cpp | 6 +- 10 files changed, 606 insertions(+), 588 deletions(-) diff --git a/hotspot/src/share/vm/ci/ciEnv.cpp b/hotspot/src/share/vm/ci/ciEnv.cpp index b95ca80d6dc..6473606d499 100644 --- a/hotspot/src/share/vm/ci/ciEnv.cpp +++ b/hotspot/src/share/vm/ci/ciEnv.cpp @@ -709,24 +709,23 @@ Method* ciEnv::lookup_method(InstanceKlass* accessor, KlassHandle h_holder(THREAD, holder); LinkResolver::check_klass_accessability(h_accessor, h_holder, KILL_COMPILE_ON_FATAL_(NULL)); methodHandle dest_method; + LinkInfo link_info(h_holder, name, sig, h_accessor, /*check_access*/true); switch (bc) { case Bytecodes::_invokestatic: dest_method = - LinkResolver::resolve_static_call_or_null(h_holder, name, sig, h_accessor); + LinkResolver::resolve_static_call_or_null(link_info); break; case Bytecodes::_invokespecial: dest_method = - LinkResolver::resolve_special_call_or_null(h_holder, name, sig, h_accessor); + LinkResolver::resolve_special_call_or_null(link_info); break; case Bytecodes::_invokeinterface: dest_method = - LinkResolver::linktime_resolve_interface_method_or_null(h_holder, name, sig, - h_accessor, true); + LinkResolver::linktime_resolve_interface_method_or_null(link_info); break; case Bytecodes::_invokevirtual: dest_method = - LinkResolver::linktime_resolve_virtual_method_or_null(h_holder, name, sig, - h_accessor, true); + LinkResolver::linktime_resolve_virtual_method_or_null(link_info); break; default: ShouldNotReachHere(); } diff --git a/hotspot/src/share/vm/ci/ciField.cpp b/hotspot/src/share/vm/ci/ciField.cpp index ba3d07157aa..9fe63de27bf 100644 --- a/hotspot/src/share/vm/ci/ciField.cpp +++ b/hotspot/src/share/vm/ci/ciField.cpp @@ -352,11 +352,11 @@ bool ciField::will_link(ciInstanceKlass* accessing_klass, } } + LinkInfo link_info(_holder->get_instanceKlass(), + _name->get_symbol(), _signature->get_symbol(), + accessing_klass->get_Klass()); fieldDescriptor result; - LinkResolver::resolve_field(result, _holder->get_instanceKlass(), - _name->get_symbol(), _signature->get_symbol(), - accessing_klass->get_Klass(), bc, true, false, - KILL_COMPILE_ON_FATAL_(false)); + LinkResolver::resolve_field(result, link_info, bc, false, KILL_COMPILE_ON_FATAL_(false)); // update the hit-cache, unless there is a problem with memory scoping: if (accessing_klass->is_shared() || !is_shared()) { diff --git a/hotspot/src/share/vm/ci/ciMethod.cpp b/hotspot/src/share/vm/ci/ciMethod.cpp index 859bd3eef4c..81ef4bf37df 100644 --- a/hotspot/src/share/vm/ci/ciMethod.cpp +++ b/hotspot/src/share/vm/ci/ciMethod.cpp @@ -1,5 +1,5 @@ /* - * Copyright (c) 1999, 2013, Oracle and/or its affiliates. All rights reserved. + * Copyright (c) 1999, 2015, 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 @@ -786,6 +786,7 @@ ciMethod* ciMethod::resolve_invoke(ciKlass* caller, ciKlass* exact_receiver, boo Symbol* h_name = name()->get_symbol(); Symbol* h_signature = signature()->get_symbol(); + LinkInfo link_info(h_resolved, h_name, h_signature, caller_klass, check_access); methodHandle m; // Only do exact lookup if receiver klass has been linked. Otherwise, // the vtable has not been setup, and the LinkResolver will fail. @@ -793,9 +794,9 @@ ciMethod* ciMethod::resolve_invoke(ciKlass* caller, ciKlass* exact_receiver, boo || InstanceKlass::cast(h_recv())->is_linked() && !exact_receiver->is_interface()) { if (holder()->is_interface()) { - m = LinkResolver::resolve_interface_call_or_null(h_recv, h_resolved, h_name, h_signature, caller_klass, check_access); + m = LinkResolver::resolve_interface_call_or_null(h_recv, link_info); } else { - m = LinkResolver::resolve_virtual_call_or_null(h_recv, h_resolved, h_name, h_signature, caller_klass, check_access); + m = LinkResolver::resolve_virtual_call_or_null(h_recv, link_info); } } @@ -839,7 +840,8 @@ int ciMethod::resolve_vtable_index(ciKlass* caller, ciKlass* receiver) { Symbol* h_name = name()->get_symbol(); Symbol* h_signature = signature()->get_symbol(); - vtable_index = LinkResolver::resolve_virtual_vtable_index(h_recv, h_recv, h_name, h_signature, caller_klass); + LinkInfo link_info(h_recv, h_name, h_signature, caller_klass); + vtable_index = LinkResolver::resolve_virtual_vtable_index(h_recv, link_info); if (vtable_index == Method::nonvirtual_vtable_index) { // A statically bound method. Return "no such index". vtable_index = Method::invalid_vtable_index; @@ -1285,10 +1287,8 @@ bool ciMethod::check_call(int refinfo_index, bool is_static) const { EXCEPTION_MARK; HandleMark hm(THREAD); constantPoolHandle pool (THREAD, get_Method()->constants()); - methodHandle spec_method; - KlassHandle spec_klass; Bytecodes::Code code = (is_static ? Bytecodes::_invokestatic : Bytecodes::_invokevirtual); - LinkResolver::resolve_method_statically(spec_method, spec_klass, code, pool, refinfo_index, THREAD); + methodHandle spec_method = LinkResolver::resolve_method_statically(code, pool, refinfo_index, THREAD); if (HAS_PENDING_EXCEPTION) { CLEAR_PENDING_EXCEPTION; return false; diff --git a/hotspot/src/share/vm/interpreter/bytecode.cpp b/hotspot/src/share/vm/interpreter/bytecode.cpp index 70c3584e55c..80b95d627b5 100644 --- a/hotspot/src/share/vm/interpreter/bytecode.cpp +++ b/hotspot/src/share/vm/interpreter/bytecode.cpp @@ -1,5 +1,5 @@ /* - * Copyright (c) 1997, 2012, Oracle and/or its affiliates. All rights reserved. + * Copyright (c) 1997, 2015, 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 @@ -147,13 +147,10 @@ BasicType Bytecode_member_ref::result_type() const { methodHandle Bytecode_invoke::static_target(TRAPS) { - methodHandle m; - KlassHandle resolved_klass; constantPoolHandle constants(THREAD, this->constants()); Bytecodes::Code bc = invoke_code(); - LinkResolver::resolve_method_statically(m, resolved_klass, bc, constants, index(), CHECK_(methodHandle())); - return m; + return LinkResolver::resolve_method_statically(bc, constants, index(), THREAD); } Handle Bytecode_invoke::appendix(TRAPS) { diff --git a/hotspot/src/share/vm/interpreter/linkResolver.cpp b/hotspot/src/share/vm/interpreter/linkResolver.cpp index 5141f19dbc2..d3a97225c20 100644 --- a/hotspot/src/share/vm/interpreter/linkResolver.cpp +++ b/hotspot/src/share/vm/interpreter/linkResolver.cpp @@ -52,13 +52,17 @@ // Implementation of CallInfo -void CallInfo::set_static(KlassHandle resolved_klass, methodHandle resolved_method, TRAPS) { +void CallInfo::set_static(KlassHandle resolved_klass, const methodHandle& resolved_method, TRAPS) { int vtable_index = Method::nonvirtual_vtable_index; set_common(resolved_klass, resolved_klass, resolved_method, resolved_method, CallInfo::direct_call, vtable_index, CHECK); } -void CallInfo::set_interface(KlassHandle resolved_klass, KlassHandle selected_klass, methodHandle resolved_method, methodHandle selected_method, int itable_index, TRAPS) { +void CallInfo::set_interface(KlassHandle resolved_klass, + KlassHandle selected_klass, + const methodHandle& resolved_method, + const methodHandle& selected_method, + int itable_index, TRAPS) { // This is only called for interface methods. If the resolved_method // comes from java/lang/Object, it can be the subject of a virtual call, so // we should pick the vtable index from the resolved method. @@ -68,7 +72,11 @@ void CallInfo::set_interface(KlassHandle resolved_klass, KlassHandle selected_kl set_common(resolved_klass, selected_klass, resolved_method, selected_method, CallInfo::itable_call, itable_index, CHECK); } -void CallInfo::set_virtual(KlassHandle resolved_klass, KlassHandle selected_klass, methodHandle resolved_method, methodHandle selected_method, int vtable_index, TRAPS) { +void CallInfo::set_virtual(KlassHandle resolved_klass, + KlassHandle selected_klass, + const methodHandle& resolved_method, + const methodHandle& selected_method, + int vtable_index, TRAPS) { assert(vtable_index >= 0 || vtable_index == Method::nonvirtual_vtable_index, "valid index"); assert(vtable_index < 0 || !resolved_method->has_vtable_index() || vtable_index == resolved_method->vtable_index(), ""); CallKind kind = (vtable_index >= 0 && !resolved_method->can_be_statically_bound() ? CallInfo::vtable_call : CallInfo::direct_call); @@ -76,7 +84,9 @@ void CallInfo::set_virtual(KlassHandle resolved_klass, KlassHandle selected_klas assert(!resolved_method->is_compiled_lambda_form(), "these must be handled via an invokehandle call"); } -void CallInfo::set_handle(methodHandle resolved_method, Handle resolved_appendix, Handle resolved_method_type, TRAPS) { +void CallInfo::set_handle(const methodHandle& resolved_method, + Handle resolved_appendix, + Handle resolved_method_type, TRAPS) { if (resolved_method.is_null()) { THROW_MSG(vmSymbols::java_lang_InternalError(), "resolved method is null"); } @@ -93,8 +103,8 @@ void CallInfo::set_handle(methodHandle resolved_method, Handle resolved_appendix void CallInfo::set_common(KlassHandle resolved_klass, KlassHandle selected_klass, - methodHandle resolved_method, - methodHandle selected_method, + const methodHandle& resolved_method, + const methodHandle& selected_method, CallKind kind, int index, TRAPS) { @@ -210,8 +220,52 @@ void CallInfo::verify() { } #endif //ASSERT +#ifndef PRODUCT +void CallInfo::print() { + ResourceMark rm; + const char* kindstr = "unknown"; + switch (_call_kind) { + case direct_call: kindstr = "direct"; break; + case vtable_call: kindstr = "vtable"; break; + case itable_call: kindstr = "itable"; break; + } + tty->print_cr("Call %s@%d %s", kindstr, _call_index, + _resolved_method.is_null() ? "(none)" : _resolved_method->name_and_sig_as_C_string()); +} +#endif +//------------------------------------------------------------------------------------------------------------------------ +// Implementation of LinkInfo +LinkInfo::LinkInfo(constantPoolHandle pool, int index, TRAPS) { + // resolve klass + Klass* result = pool->klass_ref_at(index, CHECK); + _resolved_klass = KlassHandle(THREAD, result); + + // Get name, signature, and static klass + _name = pool->name_ref_at(index); + _signature = pool->signature_ref_at(index); + _current_klass = KlassHandle(THREAD, pool->pool_holder()); + + // Coming from the constant pool always checks access + _check_access = true; +} + +char* LinkInfo::method_string() const { + return Method::name_and_sig_as_C_string(_resolved_klass(), _name, _signature); +} + +#ifndef PRODUCT +void LinkInfo::print() { + ResourceMark rm; + tty->print_cr("Link resolved_klass=%s name=%s signature=%s current_klass=%s check_access=%s", + _resolved_klass->name()->as_C_string(), + _name->as_C_string(), + _signature->as_C_string(), + _current_klass.is_null() ? "(none)" : _current_klass->name()->as_C_string(), + _check_access ? "true" : "false"); +} +#endif // PRODUCT //------------------------------------------------------------------------------------------------------------------------ // Klass resolution @@ -231,11 +285,6 @@ void LinkResolver::check_klass_accessability(KlassHandle ref_klass, KlassHandle } } -void LinkResolver::resolve_klass(KlassHandle& result, constantPoolHandle pool, int index, TRAPS) { - Klass* result_oop = pool->klass_ref_at(index, CHECK); - result = KlassHandle(THREAD, result_oop); -} - //------------------------------------------------------------------------------------------------------------------------ // Method resolution // @@ -243,76 +292,84 @@ void LinkResolver::resolve_klass(KlassHandle& result, constantPoolHandle pool, i // Look up method in klasses, including static methods // Then look up local default methods -void LinkResolver::lookup_method_in_klasses(methodHandle& result, KlassHandle klass, Symbol* name, Symbol* signature, bool checkpolymorphism, bool in_imethod_resolve, TRAPS) { +methodHandle LinkResolver::lookup_method_in_klasses(const LinkInfo& link_info, + bool checkpolymorphism, + bool in_imethod_resolve, TRAPS) { + KlassHandle klass = link_info.resolved_klass(); + Symbol* name = link_info.name(); + Symbol* signature = link_info.signature(); + // Ignore overpasses so statics can be found during resolution - Method* result_oop = klass->uncached_lookup_method(name, signature, Klass::skip_overpass); + Method* result = klass->uncached_lookup_method(name, signature, Klass::skip_overpass); if (klass->oop_is_array()) { // Only consider klass and super klass for arrays - result = methodHandle(THREAD, result_oop); - return; + return methodHandle(THREAD, result); } // JDK 8, JVMS 5.4.3.4: Interface method resolution should // ignore static and non-public methods of java.lang.Object, // like clone, finalize, registerNatives. if (in_imethod_resolve && - result_oop != NULL && + result != NULL && klass->is_interface() && - (result_oop->is_static() || !result_oop->is_public()) && - result_oop->method_holder() == SystemDictionary::Object_klass()) { - result_oop = NULL; + (result->is_static() || !result->is_public()) && + result->method_holder() == SystemDictionary::Object_klass()) { + result = NULL; } // Before considering default methods, check for an overpass in the // current class if a method has not been found. - if (result_oop == NULL) { - result_oop = InstanceKlass::cast(klass())->find_method(name, signature); + if (result == NULL) { + result = InstanceKlass::cast(klass())->find_method(name, signature); } - if (result_oop == NULL) { + if (result == NULL) { Array* default_methods = InstanceKlass::cast(klass())->default_methods(); if (default_methods != NULL) { - result_oop = InstanceKlass::find_method(default_methods, name, signature); + result = InstanceKlass::find_method(default_methods, name, signature); } } - if (checkpolymorphism && result_oop != NULL) { - vmIntrinsics::ID iid = result_oop->intrinsic_id(); + if (checkpolymorphism && result != NULL) { + vmIntrinsics::ID iid = result->intrinsic_id(); if (MethodHandles::is_signature_polymorphic(iid)) { // Do not link directly to these. The VM must produce a synthetic one using lookup_polymorphic_method. - return; + return NULL; } } - result = methodHandle(THREAD, result_oop); + return methodHandle(THREAD, result); } // returns first instance method // Looks up method in classes, then looks up local default methods -void LinkResolver::lookup_instance_method_in_klasses(methodHandle& result, KlassHandle klass, Symbol* name, Symbol* signature, TRAPS) { - Method* result_oop = klass->uncached_lookup_method(name, signature, Klass::find_overpass); - result = methodHandle(THREAD, result_oop); - while (!result.is_null() && result->is_static() && result->method_holder()->super() != NULL) { - KlassHandle super_klass = KlassHandle(THREAD, result->method_holder()->super()); - result = methodHandle(THREAD, super_klass->uncached_lookup_method(name, signature, Klass::find_overpass)); +methodHandle LinkResolver::lookup_instance_method_in_klasses(KlassHandle klass, + Symbol* name, + Symbol* signature, TRAPS) { + Method* result = klass->uncached_lookup_method(name, signature, Klass::find_overpass); + + while (result != NULL && result->is_static() && result->method_holder()->super() != NULL) { + Klass* super_klass = result->method_holder()->super(); + result = super_klass->uncached_lookup_method(name, signature, Klass::find_overpass); } if (klass->oop_is_array()) { // Only consider klass and super klass for arrays - return; + return methodHandle(THREAD, result); } - if (result.is_null()) { + if (result == NULL) { Array* default_methods = InstanceKlass::cast(klass())->default_methods(); if (default_methods != NULL) { - result = methodHandle(InstanceKlass::find_method(default_methods, name, signature)); - assert(result.is_null() || !result->is_static(), "static defaults not allowed"); + result = InstanceKlass::find_method(default_methods, name, signature); + assert(result == NULL || !result->is_static(), "static defaults not allowed"); } } + return methodHandle(THREAD, result); } int LinkResolver::vtable_index_of_interface_method(KlassHandle klass, - methodHandle resolved_method) { + const methodHandle& resolved_method) { int vtable_index = Method::invalid_vtable_index; Symbol* name = resolved_method->name(); @@ -336,21 +393,26 @@ int LinkResolver::vtable_index_of_interface_method(KlassHandle klass, return vtable_index; } -void LinkResolver::lookup_method_in_interfaces(methodHandle& result, KlassHandle klass, Symbol* name, Symbol* signature, TRAPS) { - InstanceKlass *ik = InstanceKlass::cast(klass()); +methodHandle LinkResolver::lookup_method_in_interfaces(const LinkInfo& cp_info, TRAPS) { + InstanceKlass *ik = InstanceKlass::cast(cp_info.resolved_klass()()); // Specify 'true' in order to skip default methods when searching the // interfaces. Function lookup_method_in_klasses() already looked for // the method in the default methods table. - result = methodHandle(THREAD, ik->lookup_method_in_all_interfaces(name, signature, Klass::skip_defaults)); + return methodHandle(THREAD, + ik->lookup_method_in_all_interfaces(cp_info.name(), cp_info.signature(), + Klass::skip_defaults)); } -void LinkResolver::lookup_polymorphic_method(methodHandle& result, - KlassHandle klass, Symbol* name, Symbol* full_signature, - KlassHandle current_klass, +methodHandle LinkResolver::lookup_polymorphic_method( + const LinkInfo& link_info, Handle *appendix_result_or_null, Handle *method_type_result, TRAPS) { + KlassHandle klass = link_info.resolved_klass(); + Symbol* name = link_info.name(); + Symbol* full_signature = link_info.signature(); + vmIntrinsics::ID iid = MethodHandles::signature_polymorphic_name_id(name); if (TraceMethodHandles) { ResourceMark rm(THREAD); @@ -365,7 +427,7 @@ void LinkResolver::lookup_polymorphic_method(methodHandle& result, // Do not erase last argument type (MemberName) if it is a static linkTo method. bool keep_last_arg = MethodHandles::is_signature_polymorphic_static(iid); TempNewSymbol basic_signature = - MethodHandles::lookup_basic_type_signature(full_signature, keep_last_arg, CHECK); + MethodHandles::lookup_basic_type_signature(full_signature, keep_last_arg, CHECK_NULL); if (TraceMethodHandles) { ResourceMark rm(THREAD); tty->print_cr("lookup_polymorphic_method %s %s => basic %s", @@ -373,9 +435,9 @@ void LinkResolver::lookup_polymorphic_method(methodHandle& result, full_signature->as_C_string(), basic_signature->as_C_string()); } - result = SystemDictionary::find_method_handle_intrinsic(iid, + methodHandle result = SystemDictionary::find_method_handle_intrinsic(iid, basic_signature, - CHECK); + CHECK_NULL); if (result.not_null()) { assert(result->is_method_handle_intrinsic(), "MH.invokeBasic or MH.linkTo* intrinsic"); assert(result->intrinsic_id() != vmIntrinsics::_invokeGeneric, "wrong place to find this"); @@ -384,8 +446,8 @@ void LinkResolver::lookup_polymorphic_method(methodHandle& result, tty->print("lookup_polymorphic_method => intrinsic "); result->print_on(tty); } - return; } + return result; } else if (iid == vmIntrinsics::_invokeGeneric && !THREAD->is_Compiler_thread() && appendix_result_or_null != NULL) { @@ -399,18 +461,19 @@ void LinkResolver::lookup_polymorphic_method(methodHandle& result, Handle(), Handle(), true, - CHECK); + CHECK_NULL); } } Handle appendix; Handle method_type; - result = SystemDictionary::find_method_handle_invoker(name, + methodHandle result = SystemDictionary::find_method_handle_invoker( + name, full_signature, - current_klass, + link_info.current_klass(), &appendix, &method_type, - CHECK); + CHECK_NULL); if (TraceMethodHandles) { tty->print("lookup_polymorphic_method => (via Java) "); result->print_on(tty); @@ -423,7 +486,7 @@ void LinkResolver::lookup_polymorphic_method(methodHandle& result, ResourceMark rm(THREAD); TempNewSymbol basic_signature = - MethodHandles::lookup_basic_type_signature(full_signature, CHECK); + MethodHandles::lookup_basic_type_signature(full_signature, CHECK_NULL); int actual_size_of_params = result->size_of_parameters(); int expected_size_of_params = ArgumentSizeComputer(basic_signature).size(); // +1 for MethodHandle.this, +1 for trailing MethodType @@ -441,16 +504,17 @@ void LinkResolver::lookup_polymorphic_method(methodHandle& result, assert(appendix_result_or_null != NULL, ""); (*appendix_result_or_null) = appendix; (*method_type_result) = method_type; - return; } + return result; } } + return NULL; } void LinkResolver::check_method_accessability(KlassHandle ref_klass, KlassHandle resolved_klass, KlassHandle sel_klass, - methodHandle sel_method, + const methodHandle& sel_method, TRAPS) { AccessFlags flags = sel_method->access_flags(); @@ -493,8 +557,8 @@ void LinkResolver::check_method_accessability(KlassHandle ref_klass, } } -void LinkResolver::resolve_method_statically(methodHandle& resolved_method, KlassHandle& resolved_klass, - Bytecodes::Code code, constantPoolHandle pool, int index, TRAPS) { +methodHandle LinkResolver::resolve_method_statically(Bytecodes::Code code, + constantPoolHandle pool, int index, TRAPS) { // This method is used only // (1) in C2 from InlineTree::ok_to_inline (via ciMethod::check_call), // and @@ -502,49 +566,108 @@ void LinkResolver::resolve_method_statically(methodHandle& resolved_method, Klas // It appears to fail when applied to an invokeinterface call site. // FIXME: Remove this method and ciMethod::check_call; refactor to use the other LinkResolver entry points. // resolve klass + KlassHandle resolved_klass; if (code == Bytecodes::_invokedynamic) { resolved_klass = SystemDictionary::MethodHandle_klass(); Symbol* method_name = vmSymbols::invoke_name(); Symbol* method_signature = pool->signature_ref_at(index); KlassHandle current_klass(THREAD, pool->pool_holder()); - resolve_method(resolved_method, resolved_klass, method_name, method_signature, current_klass, true, false, CHECK); - return; + LinkInfo link_info(resolved_klass, method_name, method_signature, current_klass); + return resolve_method(link_info, /*require_methodref*/false, THREAD); } - resolve_klass(resolved_klass, pool, index, CHECK); - - Symbol* method_name = pool->name_ref_at(index); - Symbol* method_signature = pool->signature_ref_at(index); - KlassHandle current_klass(THREAD, pool->pool_holder()); + LinkInfo link_info(pool, index, CHECK_NULL); + resolved_klass = link_info.resolved_klass(); if (pool->has_preresolution() || (resolved_klass() == SystemDictionary::MethodHandle_klass() && - MethodHandles::is_signature_polymorphic_name(resolved_klass(), method_name))) { - Method* result_oop = ConstantPool::method_at_if_loaded(pool, index); - if (result_oop != NULL) { - resolved_method = methodHandle(THREAD, result_oop); - return; + MethodHandles::is_signature_polymorphic_name(resolved_klass(), link_info.name()))) { + Method* result = ConstantPool::method_at_if_loaded(pool, index); + if (result != NULL) { + return methodHandle(THREAD, result); } } if (code == Bytecodes::_invokeinterface) { - resolve_interface_method(resolved_method, resolved_klass, method_name, method_signature, current_klass, true, true, CHECK); + return resolve_interface_method(link_info, true, THREAD); } else if (code == Bytecodes::_invokevirtual) { - resolve_method(resolved_method, resolved_klass, method_name, method_signature, current_klass, true, true, CHECK); + return resolve_method(link_info, /*require_methodref*/true, THREAD); } else if (!resolved_klass->is_interface()) { - resolve_method(resolved_method, resolved_klass, method_name, method_signature, current_klass, true, false, CHECK); + return resolve_method(link_info, /*require_methodref*/false, THREAD); } else { bool nostatics = (code == Bytecodes::_invokestatic) ? false : true; - resolve_interface_method(resolved_method, resolved_klass, method_name, method_signature, current_klass, true, nostatics, CHECK); + return resolve_interface_method(link_info, nostatics, THREAD); } } -void LinkResolver::resolve_method(methodHandle& resolved_method, KlassHandle resolved_klass, - Symbol* method_name, Symbol* method_signature, - KlassHandle current_klass, bool check_access, - bool require_methodref, TRAPS) { +// Check and print a loader constraint violation message for method or interface method +void LinkResolver::check_method_loader_constraints(const LinkInfo& link_info, + const methodHandle& resolved_method, + const char* method_type, TRAPS) { + Handle current_loader(THREAD, link_info.current_klass()->class_loader()); + Handle resolved_loader(THREAD, resolved_method->method_holder()->class_loader()); + + ResourceMark rm(THREAD); + Symbol* failed_type_symbol = + SystemDictionary::check_signature_loaders(link_info.signature(), current_loader, + resolved_loader, true, CHECK); + if (failed_type_symbol != NULL) { + const char* msg = "loader constraint violation: when resolving %s" + " \"%s\" the class loader (instance of %s) of the current class, %s," + " and the class loader (instance of %s) for the method's defining class, %s, have" + " different Class objects for the type %s used in the signature"; + char* sig = link_info.method_string(); + const char* loader1_name = SystemDictionary::loader_name(current_loader()); + char* current = link_info.current_klass()->name()->as_C_string(); + const char* loader2_name = SystemDictionary::loader_name(resolved_loader()); + char* target = resolved_method->method_holder()->name()->as_C_string(); + char* failed_type_name = failed_type_symbol->as_C_string(); + size_t buflen = strlen(msg) + strlen(sig) + strlen(loader1_name) + + strlen(current) + strlen(loader2_name) + strlen(target) + + strlen(failed_type_name) + strlen(method_type) + 1; + char* buf = NEW_RESOURCE_ARRAY_IN_THREAD(THREAD, char, buflen); + jio_snprintf(buf, buflen, msg, method_type, sig, loader1_name, current, loader2_name, + target, failed_type_name); + THROW_MSG(vmSymbols::java_lang_LinkageError(), buf); + } +} + +void LinkResolver::check_field_loader_constraints(Symbol* field, Symbol* sig, + KlassHandle current_klass, + KlassHandle sel_klass, TRAPS) { + Handle ref_loader(THREAD, current_klass->class_loader()); + Handle sel_loader(THREAD, sel_klass->class_loader()); + + ResourceMark rm(THREAD); // needed for check_signature_loaders + Symbol* failed_type_symbol = + SystemDictionary::check_signature_loaders(sig, + ref_loader, sel_loader, + false, + CHECK); + if (failed_type_symbol != NULL) { + const char* msg = "loader constraint violation: when resolving field" + " \"%s\" the class loader (instance of %s) of the referring class, " + "%s, and the class loader (instance of %s) for the field's resolved " + "type, %s, have different Class objects for that type"; + char* field_name = field->as_C_string(); + const char* loader1_name = SystemDictionary::loader_name(ref_loader()); + char* sel = sel_klass->name()->as_C_string(); + const char* loader2_name = SystemDictionary::loader_name(sel_loader()); + char* failed_type_name = failed_type_symbol->as_C_string(); + size_t buflen = strlen(msg) + strlen(field_name) + strlen(loader1_name) + + strlen(sel) + strlen(loader2_name) + strlen(failed_type_name) + 1; + char* buf = NEW_RESOURCE_ARRAY_IN_THREAD(THREAD, char, buflen); + jio_snprintf(buf, buflen, msg, field_name, loader1_name, sel, loader2_name, + failed_type_name); + THROW_MSG(vmSymbols::java_lang_LinkageError(), buf); + } +} + +methodHandle LinkResolver::resolve_method(const LinkInfo& link_info, + bool require_methodref, TRAPS) { Handle nested_exception; + KlassHandle resolved_klass = link_info.resolved_klass(); // 1. check if methodref required, that resolved_klass is not interfacemethodref if (require_methodref && resolved_klass->is_interface()) { @@ -552,20 +675,19 @@ void LinkResolver::resolve_method(methodHandle& resolved_method, KlassHandle res char buf[200]; jio_snprintf(buf, sizeof(buf), "Found interface %s, but class was expected", resolved_klass()->external_name()); - THROW_MSG(vmSymbols::java_lang_IncompatibleClassChangeError(), buf); + THROW_MSG_NULL(vmSymbols::java_lang_IncompatibleClassChangeError(), buf); } // 2. lookup method in resolved klass and its super klasses - lookup_method_in_klasses(resolved_method, resolved_klass, method_name, method_signature, true, false, CHECK); + methodHandle resolved_method = lookup_method_in_klasses(link_info, true, false, CHECK_NULL); if (resolved_method.is_null() && !resolved_klass->oop_is_array()) { // not found in the class hierarchy // 3. lookup method in all the interfaces implemented by the resolved klass - lookup_method_in_interfaces(resolved_method, resolved_klass, method_name, method_signature, CHECK); + resolved_method = lookup_method_in_interfaces(link_info, CHECK_NULL); if (resolved_method.is_null()) { // JSR 292: see if this is an implicitly generated method MethodHandle.linkToVirtual(*...), etc - lookup_polymorphic_method(resolved_method, resolved_klass, method_name, method_signature, - current_klass, (Handle*)NULL, (Handle*)NULL, THREAD); + resolved_method = lookup_polymorphic_method(link_info, (Handle*)NULL, (Handle*)NULL, THREAD); if (HAS_PENDING_EXCEPTION) { nested_exception = Handle(THREAD, PENDING_EXCEPTION); CLEAR_PENDING_EXCEPTION; @@ -576,15 +698,16 @@ void LinkResolver::resolve_method(methodHandle& resolved_method, KlassHandle res if (resolved_method.is_null()) { // 4. method lookup failed ResourceMark rm(THREAD); - THROW_MSG_CAUSE(vmSymbols::java_lang_NoSuchMethodError(), + THROW_MSG_CAUSE_(vmSymbols::java_lang_NoSuchMethodError(), Method::name_and_sig_as_C_string(resolved_klass(), - method_name, - method_signature), - nested_exception); + link_info.name(), + link_info.signature()), + nested_exception, NULL); } // 5. access checks, access checking may be turned off when calling from within the VM. - if (check_access) { + KlassHandle current_klass = link_info.current_klass(); + if (link_info.check_access()) { assert(current_klass.not_null() , "current_klass should not be null"); // check if method can be accessed by the referring class @@ -592,76 +715,50 @@ void LinkResolver::resolve_method(methodHandle& resolved_method, KlassHandle res resolved_klass, KlassHandle(THREAD, resolved_method->method_holder()), resolved_method, - CHECK); + CHECK_NULL); // check loader constraints - Handle loader (THREAD, InstanceKlass::cast(current_klass())->class_loader()); - Handle class_loader (THREAD, resolved_method->method_holder()->class_loader()); - { - ResourceMark rm(THREAD); - Symbol* failed_type_symbol = - SystemDictionary::check_signature_loaders(method_signature, loader, - class_loader, true, CHECK); - if (failed_type_symbol != NULL) { - const char* msg = "loader constraint violation: when resolving method" - " \"%s\" the class loader (instance of %s) of the current class, %s," - " and the class loader (instance of %s) for the method's defining class, %s, have" - " different Class objects for the type %s used in the signature"; - char* sig = Method::name_and_sig_as_C_string(resolved_klass(),method_name,method_signature); - const char* loader1 = SystemDictionary::loader_name(loader()); - char* current = InstanceKlass::cast(current_klass())->name()->as_C_string(); - const char* loader2 = SystemDictionary::loader_name(class_loader()); - char* target = InstanceKlass::cast(resolved_method->method_holder()) - ->name()->as_C_string(); - char* failed_type_name = failed_type_symbol->as_C_string(); - size_t buflen = strlen(msg) + strlen(sig) + strlen(loader1) + - strlen(current) + strlen(loader2) + strlen(target) + - strlen(failed_type_name) + 1; - char* buf = NEW_RESOURCE_ARRAY_IN_THREAD(THREAD, char, buflen); - jio_snprintf(buf, buflen, msg, sig, loader1, current, loader2, - target, failed_type_name); - THROW_MSG(vmSymbols::java_lang_LinkageError(), buf); - } - } + check_method_loader_constraints(link_info, resolved_method, "method", CHECK_NULL); } + + return resolved_method; } -void LinkResolver::resolve_interface_method(methodHandle& resolved_method, - KlassHandle resolved_klass, - Symbol* method_name, - Symbol* method_signature, - KlassHandle current_klass, - bool check_access, - bool nostatics, TRAPS) { +methodHandle LinkResolver::resolve_interface_method(const LinkInfo& link_info, + bool nostatics, TRAPS) { + + KlassHandle resolved_klass = link_info.resolved_klass(); // check if klass is interface if (!resolved_klass->is_interface()) { ResourceMark rm(THREAD); char buf[200]; jio_snprintf(buf, sizeof(buf), "Found class %s, but interface was expected", resolved_klass()->external_name()); - THROW_MSG(vmSymbols::java_lang_IncompatibleClassChangeError(), buf); + THROW_MSG_NULL(vmSymbols::java_lang_IncompatibleClassChangeError(), buf); } // lookup method in this interface or its super, java.lang.Object // JDK8: also look for static methods - lookup_method_in_klasses(resolved_method, resolved_klass, method_name, method_signature, false, true, CHECK); + methodHandle resolved_method = lookup_method_in_klasses(link_info, false, true, CHECK_NULL); if (resolved_method.is_null() && !resolved_klass->oop_is_array()) { // lookup method in all the super-interfaces - lookup_method_in_interfaces(resolved_method, resolved_klass, method_name, method_signature, CHECK); + resolved_method = lookup_method_in_interfaces(link_info, CHECK_NULL); } if (resolved_method.is_null()) { // no method found ResourceMark rm(THREAD); - THROW_MSG(vmSymbols::java_lang_NoSuchMethodError(), - Method::name_and_sig_as_C_string(resolved_klass(), - method_name, - method_signature)); + THROW_MSG_NULL(vmSymbols::java_lang_NoSuchMethodError(), + Method::name_and_sig_as_C_string(resolved_klass(), + link_info.name(), + link_info.signature())); } - if (check_access) { + if (link_info.check_access()) { // JDK8 adds non-public interface methods, and accessability check requirement + KlassHandle current_klass = link_info.current_klass(); + assert(current_klass.not_null() , "current_klass should not be null"); // check if method can be accessed by the referring class @@ -669,38 +766,9 @@ void LinkResolver::resolve_interface_method(methodHandle& resolved_method, resolved_klass, KlassHandle(THREAD, resolved_method->method_holder()), resolved_method, - CHECK); + CHECK_NULL); - HandleMark hm(THREAD); - Handle loader (THREAD, InstanceKlass::cast(current_klass())->class_loader()); - Handle class_loader (THREAD, resolved_method->method_holder()->class_loader()); - { - ResourceMark rm(THREAD); - Symbol* failed_type_symbol = - SystemDictionary::check_signature_loaders(method_signature, loader, - class_loader, true, CHECK); - if (failed_type_symbol != NULL) { - const char* msg = "loader constraint violation: when resolving " - "interface method \"%s\" the class loader (instance of %s) of the " - "current class, %s, and the class loader (instance of %s) for " - "the method's defining class, %s, have different Class objects for the type %s " - "used in the signature"; - char* sig = Method::name_and_sig_as_C_string(resolved_klass(),method_name,method_signature); - const char* loader1 = SystemDictionary::loader_name(loader()); - char* current = InstanceKlass::cast(current_klass())->name()->as_C_string(); - const char* loader2 = SystemDictionary::loader_name(class_loader()); - char* target = InstanceKlass::cast(resolved_method->method_holder()) - ->name()->as_C_string(); - char* failed_type_name = failed_type_symbol->as_C_string(); - size_t buflen = strlen(msg) + strlen(sig) + strlen(loader1) + - strlen(current) + strlen(loader2) + strlen(target) + - strlen(failed_type_name) + 1; - char* buf = NEW_RESOURCE_ARRAY_IN_THREAD(THREAD, char, buflen); - jio_snprintf(buf, buflen, msg, sig, loader1, current, loader2, - target, failed_type_name); - THROW_MSG(vmSymbols::java_lang_LinkageError(), buf); - } - } + check_method_loader_constraints(link_info, resolved_method, "interface method", CHECK_NULL); } if (nostatics && resolved_method->is_static()) { @@ -709,28 +777,16 @@ void LinkResolver::resolve_interface_method(methodHandle& resolved_method, jio_snprintf(buf, sizeof(buf), "Expected instance not static method %s", Method::name_and_sig_as_C_string(resolved_klass(), resolved_method->name(), resolved_method->signature())); - THROW_MSG(vmSymbols::java_lang_IncompatibleClassChangeError(), buf); + THROW_MSG_NULL(vmSymbols::java_lang_IncompatibleClassChangeError(), buf); } if (TraceItables && Verbose) { - ResourceMark rm(THREAD); - tty->print("invokeinterface resolved method: caller-class:%s, compile-time-class:%s, method:%s, method_holder:%s, access_flags: ", - (current_klass.is_null() ? "" : current_klass->internal_name()), - (resolved_klass.is_null() ? "" : resolved_klass->internal_name()), - Method::name_and_sig_as_C_string(resolved_klass(), - resolved_method->name(), - resolved_method->signature()), - resolved_method->method_holder()->internal_name() - ); - resolved_method->access_flags().print_on(tty); - if (resolved_method->is_default_method()) { - tty->print("default "); - } - if (resolved_method->is_overpass()) { - tty->print("overpass"); - } + trace_method_resolution("invokeinterface resolved method: caller-class", + link_info.current_klass(), resolved_klass, resolved_method); tty->cr(); } + + return resolved_method; } //------------------------------------------------------------------------------------------------------------------------ @@ -739,7 +795,7 @@ void LinkResolver::resolve_interface_method(methodHandle& resolved_method, void LinkResolver::check_field_accessability(KlassHandle ref_klass, KlassHandle resolved_klass, KlassHandle sel_klass, - fieldDescriptor& fd, + const fieldDescriptor& fd, TRAPS) { if (!Reflection::verify_field_access(ref_klass(), resolved_klass(), @@ -759,30 +815,27 @@ void LinkResolver::check_field_accessability(KlassHandle ref_klass, } } -void LinkResolver::resolve_field_access(fieldDescriptor& result, constantPoolHandle pool, int index, Bytecodes::Code byte, TRAPS) { - // Load these early in case the resolve of the containing klass fails - Symbol* field = pool->name_ref_at(index); - Symbol* sig = pool->signature_ref_at(index); - - // resolve specified klass - KlassHandle resolved_klass; - resolve_klass(resolved_klass, pool, index, CHECK); - - KlassHandle current_klass(THREAD, pool->pool_holder()); - resolve_field(result, resolved_klass, field, sig, current_klass, byte, true, true, CHECK); +void LinkResolver::resolve_field_access(fieldDescriptor& fd, constantPoolHandle pool, int index, Bytecodes::Code byte, TRAPS) { + LinkInfo link_info(pool, index, CHECK); + resolve_field(fd, link_info, byte, true, CHECK); } -void LinkResolver::resolve_field(fieldDescriptor& fd, KlassHandle resolved_klass, Symbol* field, Symbol* sig, - KlassHandle current_klass, Bytecodes::Code byte, bool check_access, bool initialize_class, +void LinkResolver::resolve_field(fieldDescriptor& fd, + const LinkInfo& link_info, + Bytecodes::Code byte, bool initialize_class, TRAPS) { assert(byte == Bytecodes::_getstatic || byte == Bytecodes::_putstatic || byte == Bytecodes::_getfield || byte == Bytecodes::_putfield || byte == Bytecodes::_nofast_getfield || byte == Bytecodes::_nofast_putfield || - (byte == Bytecodes::_nop && !check_access), "bad field access bytecode"); + (byte == Bytecodes::_nop && !link_info.check_access()), "bad field access bytecode"); bool is_static = (byte == Bytecodes::_getstatic || byte == Bytecodes::_putstatic); bool is_put = (byte == Bytecodes::_putfield || byte == Bytecodes::_putstatic || byte == Bytecodes::_nofast_putfield); // Check if there's a resolved klass containing the field + KlassHandle resolved_klass = link_info.resolved_klass(); + Symbol* field = link_info.name(); + Symbol* sig = link_info.signature(); + if (resolved_klass.is_null()) { ResourceMark rm(THREAD); THROW_MSG(vmSymbols::java_lang_NoSuchFieldError(), field->as_C_string()); @@ -796,11 +849,12 @@ void LinkResolver::resolve_field(fieldDescriptor& fd, KlassHandle resolved_klass THROW_MSG(vmSymbols::java_lang_NoSuchFieldError(), field->as_C_string()); } - if (!check_access) + if (!link_info.check_access()) // Access checking may be turned off when calling from within the VM. return; // check access + KlassHandle current_klass = link_info.current_klass(); check_field_accessability(current_klass, resolved_klass, sel_klass, fd, CHECK); // check for errors @@ -827,34 +881,7 @@ void LinkResolver::resolve_field(fieldDescriptor& fd, KlassHandle resolved_klass } if (sel_klass() != current_klass()) { - HandleMark hm(THREAD); - Handle ref_loader (THREAD, InstanceKlass::cast(current_klass())->class_loader()); - Handle sel_loader (THREAD, InstanceKlass::cast(sel_klass())->class_loader()); - { - ResourceMark rm(THREAD); - Symbol* failed_type_symbol = - SystemDictionary::check_signature_loaders(sig, - ref_loader, sel_loader, - false, - CHECK); - if (failed_type_symbol != NULL) { - const char* msg = "loader constraint violation: when resolving field" - " \"%s\" the class loader (instance of %s) of the referring class, " - "%s, and the class loader (instance of %s) for the field's resolved " - "type, %s, have different Class objects for that type"; - char* field_name = field->as_C_string(); - const char* loader1 = SystemDictionary::loader_name(ref_loader()); - char* sel = InstanceKlass::cast(sel_klass())->name()->as_C_string(); - const char* loader2 = SystemDictionary::loader_name(sel_loader()); - char* failed_type_name = failed_type_symbol->as_C_string(); - size_t buflen = strlen(msg) + strlen(field_name) + strlen(loader1) + - strlen(sel) + strlen(loader2) + strlen(failed_type_name) + 1; - char* buf = NEW_RESOURCE_ARRAY_IN_THREAD(THREAD, char, buflen); - jio_snprintf(buf, buflen, msg, field_name, loader1, sel, loader2, - failed_type_name); - THROW_MSG(vmSymbols::java_lang_LinkageError(), buf); - } - } + check_field_loader_constraints(field, sig, current_klass, sel_klass, CHECK); } // return information. note that the klass is set to the actual klass containing the @@ -873,32 +900,38 @@ void LinkResolver::resolve_field(fieldDescriptor& fd, KlassHandle resolved_klass // recv_klass the receiver klass -void LinkResolver::resolve_static_call(CallInfo& result, KlassHandle& resolved_klass, Symbol* method_name, - Symbol* method_signature, KlassHandle current_klass, - bool check_access, bool initialize_class, TRAPS) { - methodHandle resolved_method; - linktime_resolve_static_method(resolved_method, resolved_klass, method_name, method_signature, current_klass, check_access, CHECK); - resolved_klass = KlassHandle(THREAD, resolved_method->method_holder()); +void LinkResolver::resolve_static_call(CallInfo& result, + const LinkInfo& link_info, + bool initialize_class, TRAPS) { + methodHandle resolved_method = linktime_resolve_static_method(link_info, CHECK); + // The resolved class can change as a result of this resolution. + KlassHandle resolved_klass = KlassHandle(THREAD, resolved_method->method_holder()); + + Method* save_resolved_method = resolved_method(); // Initialize klass (this should only happen if everything is ok) if (initialize_class && resolved_klass->should_be_initialized()) { resolved_klass->initialize(CHECK); - linktime_resolve_static_method(resolved_method, resolved_klass, method_name, method_signature, current_klass, check_access, CHECK); + // Use updated LinkInfo (to reresolve with resolved_klass as method_holder?) + LinkInfo new_info(resolved_klass, link_info.name(), link_info.signature(), + link_info.current_klass(), link_info.check_access()); + resolved_method = linktime_resolve_static_method(new_info, CHECK); } + assert(save_resolved_method == resolved_method(), "does this change?"); // setup result result.set_static(resolved_klass, resolved_method, CHECK); } // throws linktime exceptions -void LinkResolver::linktime_resolve_static_method(methodHandle& resolved_method, KlassHandle resolved_klass, - Symbol* method_name, Symbol* method_signature, - KlassHandle current_klass, bool check_access, TRAPS) { +methodHandle LinkResolver::linktime_resolve_static_method(const LinkInfo& link_info, TRAPS) { + KlassHandle resolved_klass = link_info.resolved_klass(); + methodHandle resolved_method; if (!resolved_klass->is_interface()) { - resolve_method(resolved_method, resolved_klass, method_name, method_signature, current_klass, check_access, false, CHECK); + resolved_method = resolve_method(link_info, /*require_methodref*/false, CHECK_NULL); } else { - resolve_interface_method(resolved_method, resolved_klass, method_name, method_signature, current_klass, check_access, false, CHECK); + resolved_method = resolve_interface_method(link_info, /*nostatics*/false, CHECK_NULL); } assert(resolved_method->name() != vmSymbols::class_initializer_name(), "should have been checked in verifier"); @@ -909,22 +942,25 @@ void LinkResolver::linktime_resolve_static_method(methodHandle& resolved_method, jio_snprintf(buf, sizeof(buf), "Expected static method %s", Method::name_and_sig_as_C_string(resolved_klass(), resolved_method->name(), resolved_method->signature())); - THROW_MSG(vmSymbols::java_lang_IncompatibleClassChangeError(), buf); + THROW_MSG_NULL(vmSymbols::java_lang_IncompatibleClassChangeError(), buf); } + return resolved_method; } -void LinkResolver::resolve_special_call(CallInfo& result, KlassHandle resolved_klass, Symbol* method_name, - Symbol* method_signature, KlassHandle current_klass, bool check_access, TRAPS) { - methodHandle resolved_method; - linktime_resolve_special_method(resolved_method, resolved_klass, method_name, method_signature, current_klass, check_access, CHECK); - runtime_resolve_special_method(result, resolved_method, resolved_klass, current_klass, check_access, CHECK); +void LinkResolver::resolve_special_call(CallInfo& result, + const LinkInfo& link_info, + TRAPS) { + methodHandle resolved_method = linktime_resolve_special_method(link_info, CHECK); + runtime_resolve_special_method(result, resolved_method, + link_info.resolved_klass(), + link_info.current_klass(), + link_info.check_access(), CHECK); } // throws linktime exceptions -void LinkResolver::linktime_resolve_special_method(methodHandle& resolved_method, KlassHandle resolved_klass, - Symbol* method_name, Symbol* method_signature, - KlassHandle current_klass, bool check_access, TRAPS) { +methodHandle LinkResolver::linktime_resolve_special_method(const LinkInfo& link_info, + TRAPS) { // Invokespecial is called for multiple special reasons: // @@ -932,11 +968,13 @@ void LinkResolver::linktime_resolve_special_method(methodHandle& resolved_method // superclass.method, which can also resolve to a default method // and the selected method is recalculated relative to the direct superclass // superinterface.method, which explicitly does not check shadowing + KlassHandle resolved_klass = link_info.resolved_klass(); + methodHandle resolved_method; if (!resolved_klass->is_interface()) { - resolve_method(resolved_method, resolved_klass, method_name, method_signature, current_klass, check_access, false, CHECK); + resolved_method = resolve_method(link_info, /*require_methodref*/false, CHECK_NULL); } else { - resolve_interface_method(resolved_method, resolved_klass, method_name, method_signature, current_klass, check_access, true, CHECK); + resolved_method = resolve_interface_method(link_info, /*nostatics*/true, CHECK_NULL); } // check if method name is , that it is found in same klass as static type @@ -951,10 +989,11 @@ void LinkResolver::linktime_resolve_special_method(methodHandle& resolved_method resolved_method->name()->as_C_string(), resolved_method->signature()->as_C_string() ); - return; + return NULL; } // check if invokespecial's interface method reference is in an indirect superinterface + KlassHandle current_klass = link_info.current_klass(); if (!current_klass.is_null() && resolved_klass->is_interface()) { Klass *klass_to_check = !InstanceKlass::cast(current_klass())->is_anonymous() ? current_klass() : @@ -973,7 +1012,7 @@ void LinkResolver::linktime_resolve_special_method(methodHandle& resolved_method resolved_method->name(), resolved_method->signature()), current_klass->external_name()); - THROW_MSG(vmSymbols::java_lang_IncompatibleClassChangeError(), buf); + THROW_MSG_NULL(vmSymbols::java_lang_IncompatibleClassChangeError(), buf); } } @@ -984,35 +1023,26 @@ void LinkResolver::linktime_resolve_special_method(methodHandle& resolved_method jio_snprintf(buf, sizeof(buf), "Expecting non-static method %s", Method::name_and_sig_as_C_string(resolved_klass(), - resolved_method->name(), - resolved_method->signature())); - THROW_MSG(vmSymbols::java_lang_IncompatibleClassChangeError(), buf); + resolved_method->name(), + resolved_method->signature())); + THROW_MSG_NULL(vmSymbols::java_lang_IncompatibleClassChangeError(), buf); } if (TraceItables && Verbose) { - ResourceMark rm(THREAD); - tty->print("invokespecial resolved method: caller-class:%s, compile-time-class:%s, method:%s, method_holder:%s, access_flags: ", - (current_klass.is_null() ? "" : current_klass->internal_name()), - (resolved_klass.is_null() ? "" : resolved_klass->internal_name()), - Method::name_and_sig_as_C_string(resolved_klass(), - resolved_method->name(), - resolved_method->signature()), - resolved_method->method_holder()->internal_name() - ); - resolved_method->access_flags().print_on(tty); - if (resolved_method->is_default_method()) { - tty->print("default "); - } - if (resolved_method->is_overpass()) { - tty->print("overpass"); - } + trace_method_resolution("invokespecial resolved method: caller-class:", + current_klass, resolved_klass, resolved_method); tty->cr(); } + + return resolved_method; } // throws runtime exceptions -void LinkResolver::runtime_resolve_special_method(CallInfo& result, methodHandle resolved_method, KlassHandle resolved_klass, - KlassHandle current_klass, bool check_access, TRAPS) { +void LinkResolver::runtime_resolve_special_method(CallInfo& result, + const methodHandle& resolved_method, + KlassHandle resolved_klass, + KlassHandle current_klass, + bool check_access, TRAPS) { // resolved method is selected method unless we have an old-style lookup // for a superclass method @@ -1037,7 +1067,7 @@ void LinkResolver::runtime_resolve_special_method(CallInfo& result, methodHandle resolved_method->name() != vmSymbols::object_initializer_name()) { // Lookup super method KlassHandle super_klass(THREAD, current_klass->super()); - lookup_instance_method_in_klasses(sel_method, super_klass, + sel_method = lookup_instance_method_in_klasses(super_klass, resolved_method->name(), resolved_method->signature(), CHECK); // check if found @@ -1066,26 +1096,13 @@ void LinkResolver::runtime_resolve_special_method(CallInfo& result, methodHandle ResourceMark rm(THREAD); THROW_MSG(vmSymbols::java_lang_AbstractMethodError(), Method::name_and_sig_as_C_string(resolved_klass(), - sel_method->name(), - sel_method->signature())); + sel_method->name(), + sel_method->signature())); } if (TraceItables && Verbose) { - ResourceMark rm(THREAD); - tty->print("invokespecial selected method: resolved-class:%s, method:%s, method_holder:%s, access_flags: ", - (resolved_klass.is_null() ? "" : resolved_klass->internal_name()), - Method::name_and_sig_as_C_string(resolved_klass(), - sel_method->name(), - sel_method->signature()), - sel_method->method_holder()->internal_name() - ); - sel_method->access_flags().print_on(tty); - if (sel_method->is_default_method()) { - tty->print("default "); - } - if (sel_method->is_overpass()) { - tty->print("overpass"); - } + trace_method_resolution("invokespecial selected method: resolved-class:", + resolved_klass, resolved_klass, sel_method); tty->cr(); } @@ -1093,25 +1110,29 @@ void LinkResolver::runtime_resolve_special_method(CallInfo& result, methodHandle result.set_static(resolved_klass, sel_method, CHECK); } -void LinkResolver::resolve_virtual_call(CallInfo& result, Handle recv, KlassHandle receiver_klass, KlassHandle resolved_klass, - Symbol* method_name, Symbol* method_signature, KlassHandle current_klass, - bool check_access, bool check_null_and_abstract, TRAPS) { - methodHandle resolved_method; - linktime_resolve_virtual_method(resolved_method, resolved_klass, method_name, method_signature, current_klass, check_access, CHECK); - runtime_resolve_virtual_method(result, resolved_method, resolved_klass, recv, receiver_klass, check_null_and_abstract, CHECK); +void LinkResolver::resolve_virtual_call(CallInfo& result, Handle recv, KlassHandle receiver_klass, + const LinkInfo& link_info, + bool check_null_and_abstract, TRAPS) { + methodHandle resolved_method = linktime_resolve_virtual_method(link_info, CHECK); + runtime_resolve_virtual_method(result, resolved_method, + link_info.resolved_klass(), + recv, receiver_klass, + check_null_and_abstract, CHECK); } // throws linktime exceptions -void LinkResolver::linktime_resolve_virtual_method(methodHandle &resolved_method, KlassHandle resolved_klass, - Symbol* method_name, Symbol* method_signature, - KlassHandle current_klass, bool check_access, TRAPS) { +methodHandle LinkResolver::linktime_resolve_virtual_method(const LinkInfo& link_info, + TRAPS) { // normal method resolution - resolve_method(resolved_method, resolved_klass, method_name, method_signature, current_klass, check_access, true, CHECK); + methodHandle resolved_method = resolve_method(link_info, /*require_methodref*/true, CHECK_NULL); assert(resolved_method->name() != vmSymbols::object_initializer_name(), "should have been checked in verifier"); assert(resolved_method->name() != vmSymbols::class_initializer_name (), "should have been checked in verifier"); // check if private interface method + KlassHandle resolved_klass = link_info.resolved_klass(); + KlassHandle current_klass = link_info.current_klass(); + if (resolved_klass->is_interface() && resolved_method->is_private()) { ResourceMark rm(THREAD); char buf[200]; @@ -1120,7 +1141,7 @@ void LinkResolver::linktime_resolve_virtual_method(methodHandle &resolved_method resolved_method->name(), resolved_method->signature()), (current_klass.is_null() ? "" : current_klass->internal_name())); - THROW_MSG(vmSymbols::java_lang_IncompatibleClassChangeError(), buf); + THROW_MSG_NULL(vmSymbols::java_lang_IncompatibleClassChangeError(), buf); } // check if not static @@ -1130,33 +1151,21 @@ void LinkResolver::linktime_resolve_virtual_method(methodHandle &resolved_method jio_snprintf(buf, sizeof(buf), "Expecting non-static method %s", Method::name_and_sig_as_C_string(resolved_klass(), resolved_method->name(), resolved_method->signature())); - THROW_MSG(vmSymbols::java_lang_IncompatibleClassChangeError(), buf); + THROW_MSG_NULL(vmSymbols::java_lang_IncompatibleClassChangeError(), buf); } if (PrintVtables && Verbose) { - ResourceMark rm(THREAD); - tty->print("invokevirtual resolved method: caller-class:%s, compile-time-class:%s, method:%s, method_holder:%s, access_flags: ", - (current_klass.is_null() ? "" : current_klass->internal_name()), - (resolved_klass.is_null() ? "" : resolved_klass->internal_name()), - Method::name_and_sig_as_C_string(resolved_klass(), - resolved_method->name(), - resolved_method->signature()), - resolved_method->method_holder()->internal_name() - ); - resolved_method->access_flags().print_on(tty); - if (resolved_method->is_default_method()) { - tty->print("default "); - } - if (resolved_method->is_overpass()) { - tty->print("overpass"); - } + trace_method_resolution("invokevirtual resolved method: caller-class:", + current_klass, resolved_klass, resolved_method); tty->cr(); } + + return resolved_method; } // throws runtime exceptions void LinkResolver::runtime_resolve_virtual_method(CallInfo& result, - methodHandle resolved_method, + const methodHandle& resolved_method, KlassHandle resolved_klass, Handle recv, KlassHandle recv_klass, @@ -1227,50 +1236,40 @@ void LinkResolver::runtime_resolve_virtual_method(CallInfo& result, } if (PrintVtables && Verbose) { - ResourceMark rm(THREAD); - tty->print("invokevirtual selected method: receiver-class:%s, resolved-class:%s, method:%s, method_holder:%s, vtable_index:%d, access_flags: ", - (recv_klass.is_null() ? "" : recv_klass->internal_name()), - (resolved_klass.is_null() ? "" : resolved_klass->internal_name()), - Method::name_and_sig_as_C_string(resolved_klass(), - resolved_method->name(), - resolved_method->signature()), - selected_method->method_holder()->internal_name(), - vtable_index - ); - selected_method->access_flags().print_on(tty); - if (selected_method->is_default_method()) { - tty->print("default "); - } - if (selected_method->is_overpass()) { - tty->print("overpass"); - } - tty->cr(); + trace_method_resolution("invokevirtual selected method: receiver-class:", + recv_klass, resolved_klass, selected_method); + tty->print_cr("vtable_index:%d", vtable_index); } // setup result result.set_virtual(resolved_klass, recv_klass, resolved_method, selected_method, vtable_index, CHECK); } -void LinkResolver::resolve_interface_call(CallInfo& result, Handle recv, KlassHandle recv_klass, KlassHandle resolved_klass, - Symbol* method_name, Symbol* method_signature, KlassHandle current_klass, - bool check_access, bool check_null_and_abstract, TRAPS) { - methodHandle resolved_method; - linktime_resolve_interface_method(resolved_method, resolved_klass, method_name, method_signature, current_klass, check_access, CHECK); - runtime_resolve_interface_method(result, resolved_method, resolved_klass, recv, recv_klass, check_null_and_abstract, CHECK); +void LinkResolver::resolve_interface_call(CallInfo& result, Handle recv, KlassHandle recv_klass, + const LinkInfo& link_info, + bool check_null_and_abstract, TRAPS) { + // throws linktime exceptions + methodHandle resolved_method = linktime_resolve_interface_method(link_info, CHECK); + runtime_resolve_interface_method(result, resolved_method,link_info.resolved_klass(), + recv, recv_klass, check_null_and_abstract, CHECK); } -// throws linktime exceptions -void LinkResolver::linktime_resolve_interface_method(methodHandle& resolved_method, KlassHandle resolved_klass, Symbol* method_name, - Symbol* method_signature, KlassHandle current_klass, bool check_access, TRAPS) { +methodHandle LinkResolver::linktime_resolve_interface_method(const LinkInfo& link_info, + TRAPS) { // normal interface method resolution - resolve_interface_method(resolved_method, resolved_klass, method_name, method_signature, current_klass, check_access, true, CHECK); - + methodHandle resolved_method = resolve_interface_method(link_info, true, CHECK_NULL); assert(resolved_method->name() != vmSymbols::object_initializer_name(), "should have been checked in verifier"); assert(resolved_method->name() != vmSymbols::class_initializer_name (), "should have been checked in verifier"); + + return resolved_method; } // throws runtime exceptions -void LinkResolver::runtime_resolve_interface_method(CallInfo& result, methodHandle resolved_method, KlassHandle resolved_klass, - Handle recv, KlassHandle recv_klass, bool check_null_and_abstract, TRAPS) { +void LinkResolver::runtime_resolve_interface_method(CallInfo& result, + const methodHandle& resolved_method, + KlassHandle resolved_klass, + Handle recv, + KlassHandle recv_klass, + bool check_null_and_abstract, TRAPS) { // check if receiver exists if (check_null_and_abstract && recv.is_null()) { THROW(vmSymbols::java_lang_NullPointerException()); @@ -1298,12 +1297,11 @@ void LinkResolver::runtime_resolve_interface_method(CallInfo& result, methodHand } // do lookup based on receiver klass - methodHandle sel_method; // This search must match the linktime preparation search for itable initialization // to correctly enforce loader constraints for interface method inheritance - lookup_instance_method_in_klasses(sel_method, recv_klass, - resolved_method->name(), - resolved_method->signature(), CHECK); + methodHandle sel_method = lookup_instance_method_in_klasses(recv_klass, + resolved_method->name(), + resolved_method->signature(), CHECK); if (sel_method.is_null() && !check_null_and_abstract) { // In theory this is a harmless placeholder value, but // in practice leaving in null affects the nsk default method tests. @@ -1314,9 +1312,9 @@ void LinkResolver::runtime_resolve_interface_method(CallInfo& result, methodHand if (sel_method.is_null()) { ResourceMark rm(THREAD); THROW_MSG(vmSymbols::java_lang_AbstractMethodError(), - Method::name_and_sig_as_C_string(recv_klass(), - resolved_method->name(), - resolved_method->signature())); + Method::name_and_sig_as_C_string(recv_klass(), + resolved_method->name(), + resolved_method->signature())); } // check access // Throw Illegal Access Error if sel_method is not public. @@ -1337,22 +1335,8 @@ void LinkResolver::runtime_resolve_interface_method(CallInfo& result, methodHand } if (TraceItables && Verbose) { - ResourceMark rm(THREAD); - tty->print("invokeinterface selected method: receiver-class:%s, resolved-class:%s, method:%s, method_holder:%s, access_flags: ", - (recv_klass.is_null() ? "" : recv_klass->internal_name()), - (resolved_klass.is_null() ? "" : resolved_klass->internal_name()), - Method::name_and_sig_as_C_string(resolved_klass(), - resolved_method->name(), - resolved_method->signature()), - sel_method->method_holder()->internal_name() - ); - sel_method->access_flags().print_on(tty); - if (sel_method->is_default_method()) { - tty->print("default "); - } - if (sel_method->is_overpass()) { - tty->print("overpass"); - } + trace_method_resolution("invokeinterface selected method: receiver-class", + recv_klass, resolved_klass, sel_method); tty->cr(); } // setup result @@ -1368,14 +1352,9 @@ void LinkResolver::runtime_resolve_interface_method(CallInfo& result, methodHand methodHandle LinkResolver::linktime_resolve_interface_method_or_null( - KlassHandle resolved_klass, - Symbol* method_name, - Symbol* method_signature, - KlassHandle current_klass, - bool check_access) { + const LinkInfo& link_info) { EXCEPTION_MARK; - methodHandle method_result; - linktime_resolve_interface_method(method_result, resolved_klass, method_name, method_signature, current_klass, check_access, THREAD); + methodHandle method_result = linktime_resolve_interface_method(link_info, THREAD); if (HAS_PENDING_EXCEPTION) { CLEAR_PENDING_EXCEPTION; return methodHandle(); @@ -1385,14 +1364,9 @@ methodHandle LinkResolver::linktime_resolve_interface_method_or_null( } methodHandle LinkResolver::linktime_resolve_virtual_method_or_null( - KlassHandle resolved_klass, - Symbol* method_name, - Symbol* method_signature, - KlassHandle current_klass, - bool check_access) { + const LinkInfo& link_info) { EXCEPTION_MARK; - methodHandle method_result; - linktime_resolve_virtual_method(method_result, resolved_klass, method_name, method_signature, current_klass, check_access, THREAD); + methodHandle method_result = linktime_resolve_virtual_method(link_info, THREAD); if (HAS_PENDING_EXCEPTION) { CLEAR_PENDING_EXCEPTION; return methodHandle(); @@ -1403,14 +1377,10 @@ methodHandle LinkResolver::linktime_resolve_virtual_method_or_null( methodHandle LinkResolver::resolve_virtual_call_or_null( KlassHandle receiver_klass, - KlassHandle resolved_klass, - Symbol* name, - Symbol* signature, - KlassHandle current_klass, - bool check_access) { + const LinkInfo& link_info) { EXCEPTION_MARK; CallInfo info; - resolve_virtual_call(info, Handle(), receiver_klass, resolved_klass, name, signature, current_klass, check_access, false, THREAD); + resolve_virtual_call(info, Handle(), receiver_klass, link_info, false, THREAD); if (HAS_PENDING_EXCEPTION) { CLEAR_PENDING_EXCEPTION; return methodHandle(); @@ -1420,14 +1390,10 @@ methodHandle LinkResolver::resolve_virtual_call_or_null( methodHandle LinkResolver::resolve_interface_call_or_null( KlassHandle receiver_klass, - KlassHandle resolved_klass, - Symbol* name, - Symbol* signature, - KlassHandle current_klass, - bool check_access) { + const LinkInfo& link_info) { EXCEPTION_MARK; CallInfo info; - resolve_interface_call(info, Handle(), receiver_klass, resolved_klass, name, signature, current_klass, check_access, false, THREAD); + resolve_interface_call(info, Handle(), receiver_klass, link_info, false, THREAD); if (HAS_PENDING_EXCEPTION) { CLEAR_PENDING_EXCEPTION; return methodHandle(); @@ -1435,15 +1401,12 @@ methodHandle LinkResolver::resolve_interface_call_or_null( return info.selected_method(); } -int LinkResolver::resolve_virtual_vtable_index( - KlassHandle receiver_klass, - KlassHandle resolved_klass, - Symbol* name, - Symbol* signature, - KlassHandle current_klass) { +int LinkResolver::resolve_virtual_vtable_index(KlassHandle receiver_klass, + const LinkInfo& link_info) { EXCEPTION_MARK; CallInfo info; - resolve_virtual_call(info, Handle(), receiver_klass, resolved_klass, name, signature, current_klass, true, false, THREAD); + resolve_virtual_call(info, Handle(), receiver_klass, link_info, + /*check_null_or_abstract*/false, THREAD); if (HAS_PENDING_EXCEPTION) { CLEAR_PENDING_EXCEPTION; return Method::invalid_vtable_index; @@ -1451,15 +1414,10 @@ int LinkResolver::resolve_virtual_vtable_index( return info.vtable_index(); } -methodHandle LinkResolver::resolve_static_call_or_null( - KlassHandle resolved_klass, - Symbol* name, - Symbol* signature, - KlassHandle current_klass, - bool check_access) { +methodHandle LinkResolver::resolve_static_call_or_null(const LinkInfo& link_info) { EXCEPTION_MARK; CallInfo info; - resolve_static_call(info, resolved_klass, name, signature, current_klass, check_access, false, THREAD); + resolve_static_call(info, link_info, /*initialize_class*/false, THREAD); if (HAS_PENDING_EXCEPTION) { CLEAR_PENDING_EXCEPTION; return methodHandle(); @@ -1467,15 +1425,10 @@ methodHandle LinkResolver::resolve_static_call_or_null( return info.selected_method(); } -methodHandle LinkResolver::resolve_special_call_or_null( - KlassHandle resolved_klass, - Symbol* name, - Symbol* signature, - KlassHandle current_klass, - bool check_access) { +methodHandle LinkResolver::resolve_special_call_or_null(const LinkInfo& link_info) { EXCEPTION_MARK; CallInfo info; - resolve_special_call(info, resolved_klass, name, signature, current_klass, check_access, THREAD); + resolve_special_call(info, link_info, THREAD); if (HAS_PENDING_EXCEPTION) { CLEAR_PENDING_EXCEPTION; return methodHandle(); @@ -1500,35 +1453,15 @@ void LinkResolver::resolve_invoke(CallInfo& result, Handle recv, constantPoolHan return; } -void LinkResolver::resolve_pool(KlassHandle& resolved_klass, Symbol*& method_name, Symbol*& method_signature, - KlassHandle& current_klass, constantPoolHandle pool, int index, TRAPS) { - // resolve klass - resolve_klass(resolved_klass, pool, index, CHECK); - - // Get name, signature, and static klass - method_name = pool->name_ref_at(index); - method_signature = pool->signature_ref_at(index); - current_klass = KlassHandle(THREAD, pool->pool_holder()); -} - - void LinkResolver::resolve_invokestatic(CallInfo& result, constantPoolHandle pool, int index, TRAPS) { - KlassHandle resolved_klass; - Symbol* method_name = NULL; - Symbol* method_signature = NULL; - KlassHandle current_klass; - resolve_pool(resolved_klass, method_name, method_signature, current_klass, pool, index, CHECK); - resolve_static_call(result, resolved_klass, method_name, method_signature, current_klass, true, true, CHECK); + LinkInfo link_info(pool, index, CHECK); + resolve_static_call(result, link_info, /*initialize_class*/true, CHECK); } void LinkResolver::resolve_invokespecial(CallInfo& result, constantPoolHandle pool, int index, TRAPS) { - KlassHandle resolved_klass; - Symbol* method_name = NULL; - Symbol* method_signature = NULL; - KlassHandle current_klass; - resolve_pool(resolved_klass, method_name, method_signature, current_klass, pool, index, CHECK); - resolve_special_call(result, resolved_klass, method_name, method_signature, current_klass, true, CHECK); + LinkInfo link_info(pool, index, CHECK); + resolve_special_call(result, link_info, CHECK); } @@ -1536,54 +1469,40 @@ void LinkResolver::resolve_invokevirtual(CallInfo& result, Handle recv, constantPoolHandle pool, int index, TRAPS) { - KlassHandle resolved_klass; - Symbol* method_name = NULL; - Symbol* method_signature = NULL; - KlassHandle current_klass; - resolve_pool(resolved_klass, method_name, method_signature, current_klass, pool, index, CHECK); + LinkInfo link_info(pool, index, CHECK); KlassHandle recvrKlass (THREAD, recv.is_null() ? (Klass*)NULL : recv->klass()); - resolve_virtual_call(result, recv, recvrKlass, resolved_klass, method_name, method_signature, current_klass, true, true, CHECK); + resolve_virtual_call(result, recv, recvrKlass, link_info, /*check_null_or_abstract*/true, CHECK); } void LinkResolver::resolve_invokeinterface(CallInfo& result, Handle recv, constantPoolHandle pool, int index, TRAPS) { - KlassHandle resolved_klass; - Symbol* method_name = NULL; - Symbol* method_signature = NULL; - KlassHandle current_klass; - resolve_pool(resolved_klass, method_name, method_signature, current_klass, pool, index, CHECK); + LinkInfo link_info(pool, index, CHECK); KlassHandle recvrKlass (THREAD, recv.is_null() ? (Klass*)NULL : recv->klass()); - resolve_interface_call(result, recv, recvrKlass, resolved_klass, method_name, method_signature, current_klass, true, true, CHECK); + resolve_interface_call(result, recv, recvrKlass, link_info, true, CHECK); } void LinkResolver::resolve_invokehandle(CallInfo& result, constantPoolHandle pool, int index, TRAPS) { // This guy is reached from InterpreterRuntime::resolve_invokehandle. - KlassHandle resolved_klass; - Symbol* method_name = NULL; - Symbol* method_signature = NULL; - KlassHandle current_klass; - resolve_pool(resolved_klass, method_name, method_signature, current_klass, pool, index, CHECK); + LinkInfo link_info(pool, index, CHECK); if (TraceMethodHandles) { ResourceMark rm(THREAD); - tty->print_cr("resolve_invokehandle %s %s", method_name->as_C_string(), method_signature->as_C_string()); + tty->print_cr("resolve_invokehandle %s %s", link_info.name()->as_C_string(), + link_info.signature()->as_C_string()); } - resolve_handle_call(result, resolved_klass, method_name, method_signature, current_klass, CHECK); + resolve_handle_call(result, link_info, CHECK); } -void LinkResolver::resolve_handle_call(CallInfo& result, KlassHandle resolved_klass, - Symbol* method_name, Symbol* method_signature, - KlassHandle current_klass, +void LinkResolver::resolve_handle_call(CallInfo& result, + const LinkInfo& link_info, TRAPS) { // JSR 292: this must be an implicitly generated method MethodHandle.invokeExact(*...) or similar - assert(resolved_klass() == SystemDictionary::MethodHandle_klass(), ""); - assert(MethodHandles::is_signature_polymorphic_name(method_name), ""); - methodHandle resolved_method; + assert(link_info.resolved_klass()() == SystemDictionary::MethodHandle_klass(), ""); + assert(MethodHandles::is_signature_polymorphic_name(link_info.name()), ""); Handle resolved_appendix; Handle resolved_method_type; - lookup_polymorphic_method(resolved_method, resolved_klass, - method_name, method_signature, - current_klass, &resolved_appendix, &resolved_method_type, CHECK); + methodHandle resolved_method = lookup_polymorphic_method(link_info, + &resolved_appendix, &resolved_method_type, CHECK); result.set_handle(resolved_method, resolved_appendix, resolved_method_type, CHECK); } @@ -1609,7 +1528,6 @@ static void wrap_invokedynamic_exception(TRAPS) { } void LinkResolver::resolve_invokedynamic(CallInfo& result, constantPoolHandle pool, int index, TRAPS) { - //resolve_pool(, method_name, method_signature, current_klass, pool, index, CHECK); Symbol* method_name = pool->name_ref_at(index); Symbol* method_signature = pool->signature_ref_at(index); KlassHandle current_klass = KlassHandle(THREAD, pool->pool_holder()); @@ -1667,19 +1585,27 @@ void LinkResolver::resolve_dynamic_call(CallInfo& result, wrap_invokedynamic_exception(CHECK); } -//------------------------------------------------------------------------------------------------------------------------ #ifndef PRODUCT - -void CallInfo::print() { +void LinkResolver::trace_method_resolution(const char* prefix, + KlassHandle klass, + KlassHandle resolved_klass, + const methodHandle& method) { ResourceMark rm; - const char* kindstr = "unknown"; - switch (_call_kind) { - case direct_call: kindstr = "direct"; break; - case vtable_call: kindstr = "vtable"; break; - case itable_call: kindstr = "itable"; break; + tty->print("%s%s, compile-time-class:%s, method:%s, method_holder:%s, access_flags: ", + prefix, + (klass.is_null() ? "" : klass->internal_name()), + (resolved_klass.is_null() ? "" : resolved_klass->internal_name()), + Method::name_and_sig_as_C_string(resolved_klass(), + method->name(), + method->signature()), + method->method_holder()->internal_name() + ); + method->access_flags().print_on(tty); + if (method->is_default_method()) { + tty->print("default "); + } + if (method->is_overpass()) { + tty->print("overpass "); } - tty->print_cr("Call %s@%d %s", kindstr, _call_index, - _resolved_method.is_null() ? "(none)" : _resolved_method->name_and_sig_as_C_string()); } - -#endif +#endif // PRODUCT diff --git a/hotspot/src/share/vm/interpreter/linkResolver.hpp b/hotspot/src/share/vm/interpreter/linkResolver.hpp index 5fa536491c9..d3fda4368ef 100644 --- a/hotspot/src/share/vm/interpreter/linkResolver.hpp +++ b/hotspot/src/share/vm/interpreter/linkResolver.hpp @@ -1,5 +1,5 @@ /* - * Copyright (c) 1997, 2013, Oracle and/or its affiliates. All rights reserved. + * Copyright (c) 1997, 2015, 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 @@ -36,7 +36,7 @@ // that method. If the info is invalid, the link has not been resolved // successfully. -class CallInfo VALUE_OBJ_CLASS_SPEC { +class CallInfo : public StackObj { public: // Ways that a method call might be selected (or not) based on receiver type. // Note that an invokevirtual instruction might be linked with no_dispatch, @@ -58,11 +58,22 @@ class CallInfo VALUE_OBJ_CLASS_SPEC { Handle _resolved_appendix; // extra argument in constant pool (if CPCE::has_appendix) Handle _resolved_method_type; // MethodType (for invokedynamic and invokehandle call sites) - void set_static( KlassHandle resolved_klass, methodHandle resolved_method , TRAPS); - void set_interface(KlassHandle resolved_klass, KlassHandle selected_klass, methodHandle resolved_method, methodHandle selected_method, int itable_index , TRAPS); - void set_virtual( KlassHandle resolved_klass, KlassHandle selected_klass, methodHandle resolved_method, methodHandle selected_method, int vtable_index , TRAPS); - void set_handle( methodHandle resolved_method, Handle resolved_appendix, Handle resolved_method_type, TRAPS); - void set_common( KlassHandle resolved_klass, KlassHandle selected_klass, methodHandle resolved_method, methodHandle selected_method, CallKind kind, int index, TRAPS); + void set_static(KlassHandle resolved_klass, const methodHandle& resolved_method, TRAPS); + void set_interface(KlassHandle resolved_klass, KlassHandle selected_klass, + const methodHandle& resolved_method, + const methodHandle& selected_method, + int itable_index, TRAPS); + void set_virtual(KlassHandle resolved_klass, KlassHandle selected_klass, + const methodHandle& resolved_method, + const methodHandle& selected_method, + int vtable_index, TRAPS); + void set_handle(const methodHandle& resolved_method, + Handle resolved_appendix, Handle resolved_method_type, TRAPS); + void set_common(KlassHandle resolved_klass, KlassHandle selected_klass, + const methodHandle& resolved_method, + const methodHandle& selected_method, + CallKind kind, + int index, TRAPS); friend class LinkResolver; @@ -113,6 +124,37 @@ class CallInfo VALUE_OBJ_CLASS_SPEC { void print() PRODUCT_RETURN; }; + +// Condensed information from constant pool to use to resolve the method or field. +// resolved_klass = specified class (i.e., static receiver class) +// current_klass = sending method holder (i.e., class containing the method +// containing the call being resolved) +class LinkInfo : public StackObj { + Symbol* _name; // extracted from JVM_CONSTANT_NameAndType + Symbol* _signature; + KlassHandle _resolved_klass; // class that the constant pool entry points to + KlassHandle _current_klass; // class that owns the constant pool + bool _check_access; + public: + LinkInfo(constantPoolHandle pool, int index, TRAPS); + // Condensed information from other call sites within the vm. + LinkInfo(KlassHandle resolved_klass, Symbol* name, Symbol* signature, + KlassHandle current_klass, bool check_access = true) : + _resolved_klass(resolved_klass), + _name(name), _signature(signature), _current_klass(current_klass), + _check_access(check_access) {} + + // accessors + Symbol* name() const { return _name; } + Symbol* signature() const { return _signature; } + KlassHandle resolved_klass() const { return _resolved_klass; } + KlassHandle current_klass() const { return _current_klass; } + bool check_access() const { return _check_access; } + char* method_string() const; + + void print() PRODUCT_RETURN; +}; + // Link information for getfield/putfield & getstatic/putstatic bytecodes // is represented using a fieldDescriptor. @@ -124,85 +166,136 @@ class LinkResolver: AllStatic { friend class klassItable; private: - static void lookup_method_in_klasses (methodHandle& result, KlassHandle klass, Symbol* name, Symbol* signature, bool checkpolymorphism, bool in_imethod_resolve, TRAPS); - static void lookup_instance_method_in_klasses (methodHandle& result, KlassHandle klass, Symbol* name, Symbol* signature, TRAPS); - static void lookup_method_in_interfaces (methodHandle& result, KlassHandle klass, Symbol* name, Symbol* signature, TRAPS); - static void lookup_polymorphic_method (methodHandle& result, KlassHandle klass, Symbol* name, Symbol* signature, - KlassHandle current_klass, Handle *appendix_result_or_null, Handle *method_type_result, TRAPS); - static void resolve_klass (KlassHandle& result, constantPoolHandle pool, int index, TRAPS); + static methodHandle lookup_method_in_klasses(const LinkInfo& link_info, + bool checkpolymorphism, + bool in_imethod_resolve, TRAPS); + static methodHandle lookup_method_in_interfaces(const LinkInfo& link_info, TRAPS); + static methodHandle lookup_polymorphic_method(const LinkInfo& link_info, + Handle *appendix_result_or_null, + Handle *method_type_result, TRAPS); + // Not Linktime so doesn't take LinkInfo + static methodHandle lookup_instance_method_in_klasses ( + KlassHandle klass, Symbol* name, Symbol* signature, TRAPS); - static void resolve_pool (KlassHandle& resolved_klass, Symbol*& method_name, Symbol*& method_signature, KlassHandle& current_klass, constantPoolHandle pool, int index, TRAPS); + // Similar loader constraint checking functions that throw + // LinkageError with descriptive message. + static void check_method_loader_constraints(const LinkInfo& link_info, + const methodHandle& resolved_method, + const char* method_type, TRAPS); + static void check_field_loader_constraints(Symbol* field, Symbol* sig, + KlassHandle current_klass, + KlassHandle sel_klass, TRAPS); - static void resolve_interface_method(methodHandle& resolved_method, KlassHandle resolved_klass, Symbol* method_name, Symbol* method_signature, KlassHandle current_klass, bool check_access, bool nostatics, TRAPS); - static void resolve_method (methodHandle& resolved_method, KlassHandle resolved_klass, Symbol* method_name, Symbol* method_signature, KlassHandle current_klass, bool check_access, bool require_methodref, TRAPS); + static methodHandle resolve_interface_method(const LinkInfo& link_info, bool nostatics, TRAPS); + static methodHandle resolve_method (const LinkInfo& link_info, bool require_methodref, TRAPS); - static void linktime_resolve_static_method (methodHandle& resolved_method, KlassHandle resolved_klass, Symbol* method_name, Symbol* method_signature, KlassHandle current_klass, bool check_access, TRAPS); - static void linktime_resolve_special_method (methodHandle& resolved_method, KlassHandle resolved_klass, Symbol* method_name, Symbol* method_signature, KlassHandle current_klass, bool check_access, TRAPS); - static void linktime_resolve_virtual_method (methodHandle &resolved_method, KlassHandle resolved_klass, Symbol* method_name, Symbol* method_signature,KlassHandle current_klass, bool check_access, TRAPS); - static void linktime_resolve_interface_method (methodHandle& resolved_method, KlassHandle resolved_klass, Symbol* method_name, Symbol* method_signature, KlassHandle current_klass, bool check_access, TRAPS); + static methodHandle linktime_resolve_static_method (const LinkInfo& link_info, TRAPS); + static methodHandle linktime_resolve_special_method (const LinkInfo& link_info, TRAPS); + static methodHandle linktime_resolve_virtual_method (const LinkInfo& link_info, TRAPS); + static methodHandle linktime_resolve_interface_method (const LinkInfo& link_info, TRAPS); - static void runtime_resolve_special_method (CallInfo& result, methodHandle resolved_method, KlassHandle resolved_klass, KlassHandle current_klass, bool check_access, TRAPS); - static void runtime_resolve_virtual_method (CallInfo& result, methodHandle resolved_method, KlassHandle resolved_klass, Handle recv, KlassHandle recv_klass, bool check_null_and_abstract, TRAPS); - static void runtime_resolve_interface_method (CallInfo& result, methodHandle resolved_method, KlassHandle resolved_klass, Handle recv, KlassHandle recv_klass, bool check_null_and_abstract, TRAPS); + static void runtime_resolve_special_method (CallInfo& result, + const methodHandle& resolved_method, + KlassHandle resolved_klass, + KlassHandle current_klass, + bool check_access, TRAPS); + static void runtime_resolve_virtual_method (CallInfo& result, + const methodHandle& resolved_method, + KlassHandle resolved_klass, + Handle recv, + KlassHandle recv_klass, + bool check_null_and_abstract, TRAPS); + static void runtime_resolve_interface_method (CallInfo& result, + const methodHandle& resolved_method, + KlassHandle resolved_klass, + Handle recv, + KlassHandle recv_klass, + bool check_null_and_abstract, TRAPS); - static void check_field_accessability (KlassHandle ref_klass, KlassHandle resolved_klass, KlassHandle sel_klass, fieldDescriptor& fd, TRAPS); - static void check_method_accessability (KlassHandle ref_klass, KlassHandle resolved_klass, KlassHandle sel_klass, methodHandle sel_method, TRAPS); + static void check_field_accessability(KlassHandle ref_klass, + KlassHandle resolved_klass, + KlassHandle sel_klass, + const fieldDescriptor& fd, TRAPS); + static void check_method_accessability(KlassHandle ref_klass, + KlassHandle resolved_klass, + KlassHandle sel_klass, + const methodHandle& sel_method, TRAPS); + // runtime resolving from constant pool + static void resolve_invokestatic (CallInfo& result, + constantPoolHandle pool, int index, TRAPS); + static void resolve_invokespecial (CallInfo& result, + constantPoolHandle pool, int index, TRAPS); + static void resolve_invokevirtual (CallInfo& result, Handle recv, + constantPoolHandle pool, int index, TRAPS); + static void resolve_invokeinterface(CallInfo& result, Handle recv, + constantPoolHandle pool, int index, TRAPS); + static void resolve_invokedynamic (CallInfo& result, + constantPoolHandle pool, int index, TRAPS); + static void resolve_invokehandle (CallInfo& result, + constantPoolHandle pool, int index, TRAPS); public: // constant pool resolving static void check_klass_accessability(KlassHandle ref_klass, KlassHandle sel_klass, TRAPS); - // static resolving calls (will not run any Java code); used only from Bytecode_invoke::static_target - static void resolve_method_statically(methodHandle& method_result, KlassHandle& klass_result, - Bytecodes::Code code, constantPoolHandle pool, int index, TRAPS); + // static resolving calls (will not run any Java code); + // used only from Bytecode_invoke::static_target + static methodHandle resolve_method_statically(Bytecodes::Code code, + constantPoolHandle pool, + int index, TRAPS); - // runtime/static resolving for fields - static void resolve_field_access(fieldDescriptor& result, constantPoolHandle pool, int index, Bytecodes::Code byte, TRAPS); - static void resolve_field(fieldDescriptor& result, KlassHandle resolved_klass, Symbol* field_name, Symbol* field_signature, - KlassHandle current_klass, Bytecodes::Code access_kind, bool check_access, bool initialize_class, TRAPS); + static void resolve_field_access(fieldDescriptor& result, + constantPoolHandle pool, + int index, Bytecodes::Code byte, TRAPS); + static void resolve_field(fieldDescriptor& result, const LinkInfo& link_info, + Bytecodes::Code access_kind, + bool initialize_class, TRAPS); - // source of access_kind codes: - static Bytecodes::Code field_access_kind(bool is_static, bool is_put) { - return (is_static - ? (is_put ? Bytecodes::_putstatic : Bytecodes::_getstatic) - : (is_put ? Bytecodes::_putfield : Bytecodes::_getfield )); - } + static void resolve_static_call (CallInfo& result, + const LinkInfo& link_info, + bool initialize_klass, TRAPS); + static void resolve_special_call (CallInfo& result, + const LinkInfo& link_info, + TRAPS); + static void resolve_virtual_call (CallInfo& result, Handle recv, KlassHandle recv_klass, + const LinkInfo& link_info, + bool check_null_and_abstract, TRAPS); + static void resolve_interface_call(CallInfo& result, Handle recv, KlassHandle recv_klass, + const LinkInfo& link_info, + bool check_null_and_abstract, TRAPS); + static void resolve_handle_call (CallInfo& result, + const LinkInfo& link_info, TRAPS); + static void resolve_dynamic_call (CallInfo& result, Handle bootstrap_specifier, + Symbol* method_name, Symbol* method_signature, + KlassHandle current_klass, TRAPS); - // runtime resolving: - // resolved_klass = specified class (i.e., static receiver class) - // current_klass = sending method holder (i.e., class containing the method containing the call being resolved) - static void resolve_static_call (CallInfo& result, KlassHandle& resolved_klass, Symbol* method_name, Symbol* method_signature, KlassHandle current_klass, bool check_access, bool initialize_klass, TRAPS); - static void resolve_special_call (CallInfo& result, KlassHandle resolved_klass, Symbol* method_name, Symbol* method_signature, KlassHandle current_klass, bool check_access, TRAPS); - static void resolve_virtual_call (CallInfo& result, Handle recv, KlassHandle recv_klass, KlassHandle resolved_klass, Symbol* method_name, Symbol* method_signature, KlassHandle current_klass, bool check_access, bool check_null_and_abstract, TRAPS); - static void resolve_interface_call(CallInfo& result, Handle recv, KlassHandle recv_klass, KlassHandle resolved_klass, Symbol* method_name, Symbol* method_signature, KlassHandle current_klass, bool check_access, bool check_null_and_abstract, TRAPS); - static void resolve_handle_call (CallInfo& result, KlassHandle resolved_klass, Symbol* method_name, Symbol* method_signature, KlassHandle current_klass, TRAPS); - static void resolve_dynamic_call (CallInfo& result, Handle bootstrap_specifier, Symbol* method_name, Symbol* method_signature, KlassHandle current_klass, TRAPS); + // same as above for compile-time resolution; but returns null handle instead of throwing + // an exception on error also, does not initialize klass (i.e., no side effects) + static methodHandle resolve_virtual_call_or_null (KlassHandle receiver_klass, + const LinkInfo& link_info); + static methodHandle resolve_interface_call_or_null(KlassHandle receiver_klass, + const LinkInfo& link_info); + static methodHandle resolve_static_call_or_null (const LinkInfo& link_info); + static methodHandle resolve_special_call_or_null (const LinkInfo& link_info); - // same as above for compile-time resolution; but returns null handle instead of throwing an exception on error - // also, does not initialize klass (i.e., no side effects) - static methodHandle resolve_virtual_call_or_null (KlassHandle receiver_klass, KlassHandle resolved_klass, Symbol* method_name, Symbol* method_signature, KlassHandle current_klass, bool check_access = true); - static methodHandle resolve_interface_call_or_null(KlassHandle receiver_klass, KlassHandle resolved_klass, Symbol* method_name, Symbol* method_signature, KlassHandle current_klass, bool check_access = true); - static methodHandle resolve_static_call_or_null (KlassHandle resolved_klass, Symbol* method_name, Symbol* method_signature, KlassHandle current_klass, bool check_access = true); - static methodHandle resolve_special_call_or_null (KlassHandle resolved_klass, Symbol* method_name, Symbol* method_signature, KlassHandle current_klass, bool check_access = true); - static int vtable_index_of_interface_method(KlassHandle klass, methodHandle resolved_method); + static int vtable_index_of_interface_method(KlassHandle klass, const methodHandle& resolved_method); // same as above for compile-time resolution; returns vtable_index if current_klass if linked - static int resolve_virtual_vtable_index (KlassHandle receiver_klass, KlassHandle resolved_klass, Symbol* method_name, Symbol* method_signature, KlassHandle current_klass); + static int resolve_virtual_vtable_index (KlassHandle receiver_klass, + const LinkInfo& link_info); // static resolving for compiler (does not throw exceptions, returns null handle if unsuccessful) - static methodHandle linktime_resolve_virtual_method_or_null (KlassHandle resolved_klass, Symbol* method_name, Symbol* method_signature, KlassHandle current_klass, bool check_access); - static methodHandle linktime_resolve_interface_method_or_null(KlassHandle resolved_klass, Symbol* method_name, Symbol* method_signature, KlassHandle current_klass, bool check_access); + static methodHandle linktime_resolve_virtual_method_or_null (const LinkInfo& link_info); + static methodHandle linktime_resolve_interface_method_or_null(const LinkInfo& link_info); // runtime resolving from constant pool - static void resolve_invokestatic (CallInfo& result, constantPoolHandle pool, int index, TRAPS); - static void resolve_invokespecial (CallInfo& result, constantPoolHandle pool, int index, TRAPS); - static void resolve_invokevirtual (CallInfo& result, Handle recv, constantPoolHandle pool, int index, TRAPS); - static void resolve_invokeinterface(CallInfo& result, Handle recv, constantPoolHandle pool, int index, TRAPS); - static void resolve_invokedynamic (CallInfo& result, constantPoolHandle pool, int index, TRAPS); - static void resolve_invokehandle (CallInfo& result, constantPoolHandle pool, int index, TRAPS); - - static void resolve_invoke (CallInfo& result, Handle recv, constantPoolHandle pool, int index, Bytecodes::Code byte, TRAPS); + static void resolve_invoke(CallInfo& result, Handle recv, + constantPoolHandle pool, int index, + Bytecodes::Code byte, TRAPS); + private: + static void trace_method_resolution(const char* prefix, KlassHandle klass, + KlassHandle resolved_klass, + const methodHandle& method) PRODUCT_RETURN; }; - #endif // SHARE_VM_INTERPRETER_LINKRESOLVER_HPP diff --git a/hotspot/src/share/vm/oops/klassVtable.cpp b/hotspot/src/share/vm/oops/klassVtable.cpp index aa16cb6dc21..54df450637b 100644 --- a/hotspot/src/share/vm/oops/klassVtable.cpp +++ b/hotspot/src/share/vm/oops/klassVtable.cpp @@ -1136,7 +1136,7 @@ void klassItable::initialize_itable_for_interface(int method_table_offset, Klass if (m->has_itable_index()) { // This search must match the runtime resolution, i.e. selection search for invokeinterface // to correctly enforce loader constraints for interface method inheritance - LinkResolver::lookup_instance_method_in_klasses(target, _klass, m->name(), m->signature(), CHECK); + target = LinkResolver::lookup_instance_method_in_klasses(_klass, m->name(), m->signature(), CHECK); } if (target == NULL || !target->is_public() || target->is_abstract()) { // Entry does not resolve. Leave it empty for AbstractMethodError. diff --git a/hotspot/src/share/vm/prims/methodHandles.cpp b/hotspot/src/share/vm/prims/methodHandles.cpp index b9cf17732a4..c54be6fd4ad 100644 --- a/hotspot/src/share/vm/prims/methodHandles.cpp +++ b/hotspot/src/share/vm/prims/methodHandles.cpp @@ -677,24 +677,24 @@ Handle MethodHandles::resolve_MemberName(Handle mname, KlassHandle caller, TRAPS case IS_METHOD: { CallInfo result; + LinkInfo link_info(defc, name, type, caller, caller.not_null()); { assert(!HAS_PENDING_EXCEPTION, ""); if (ref_kind == JVM_REF_invokeStatic) { LinkResolver::resolve_static_call(result, - defc, name, type, caller, caller.not_null(), false, THREAD); + link_info, false, THREAD); } else if (ref_kind == JVM_REF_invokeInterface) { LinkResolver::resolve_interface_call(result, Handle(), defc, - defc, name, type, caller, caller.not_null(), false, THREAD); + link_info, false, THREAD); } else if (mh_invoke_id != vmIntrinsics::_none) { assert(!is_signature_polymorphic_static(mh_invoke_id), ""); - LinkResolver::resolve_handle_call(result, - defc, name, type, caller, THREAD); + LinkResolver::resolve_handle_call(result, link_info, THREAD); } else if (ref_kind == JVM_REF_invokeSpecial) { LinkResolver::resolve_special_call(result, - defc, name, type, caller, caller.not_null(), THREAD); + link_info, THREAD); } else if (ref_kind == JVM_REF_invokeVirtual) { LinkResolver::resolve_virtual_call(result, Handle(), defc, - defc, name, type, caller, caller.not_null(), false, THREAD); + link_info, false, THREAD); } else { assert(false, err_msg("ref_kind=%d", ref_kind)); } @@ -714,11 +714,11 @@ Handle MethodHandles::resolve_MemberName(Handle mname, KlassHandle caller, TRAPS case IS_CONSTRUCTOR: { CallInfo result; + LinkInfo link_info(defc, name, type, caller, caller.not_null()); { assert(!HAS_PENDING_EXCEPTION, ""); if (name == vmSymbols::object_initializer_name()) { - LinkResolver::resolve_special_call(result, - defc, name, type, caller, caller.not_null(), THREAD); + LinkResolver::resolve_special_call(result, link_info, THREAD); } else { break; // will throw after end of switch } @@ -735,7 +735,8 @@ Handle MethodHandles::resolve_MemberName(Handle mname, KlassHandle caller, TRAPS fieldDescriptor result; // find_field initializes fd if found { assert(!HAS_PENDING_EXCEPTION, ""); - LinkResolver::resolve_field(result, defc, name, type, caller, Bytecodes::_nop, false, false, THREAD); + LinkInfo link_info(defc, name, type, caller, /*check_access*/false); + LinkResolver::resolve_field(result, link_info, Bytecodes::_nop, false, THREAD); if (HAS_PENDING_EXCEPTION) { return empty; } diff --git a/hotspot/src/share/vm/runtime/javaCalls.cpp b/hotspot/src/share/vm/runtime/javaCalls.cpp index a7bdc680e93..9d4eccd8d69 100644 --- a/hotspot/src/share/vm/runtime/javaCalls.cpp +++ b/hotspot/src/share/vm/runtime/javaCalls.cpp @@ -1,5 +1,5 @@ /* - * Copyright (c) 1997, 2014, Oracle and/or its affiliates. All rights reserved. + * Copyright (c) 1997, 2015, 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 @@ -179,9 +179,9 @@ void JavaCalls::call_virtual(JavaValue* result, KlassHandle spec_klass, Symbol* CallInfo callinfo; Handle receiver = args->receiver(); KlassHandle recvrKlass(THREAD, receiver.is_null() ? (Klass*)NULL : receiver->klass()); + LinkInfo link_info(spec_klass, name, signature, KlassHandle(), /*check_access*/false); LinkResolver::resolve_virtual_call( - callinfo, receiver, recvrKlass, spec_klass, name, signature, - KlassHandle(), false, true, CHECK); + callinfo, receiver, recvrKlass, link_info, true, CHECK); methodHandle method = callinfo.selected_method(); assert(method.not_null(), "should have thrown exception"); @@ -216,7 +216,8 @@ void JavaCalls::call_virtual(JavaValue* result, Handle receiver, KlassHandle spe void JavaCalls::call_special(JavaValue* result, KlassHandle klass, Symbol* name, Symbol* signature, JavaCallArguments* args, TRAPS) { CallInfo callinfo; - LinkResolver::resolve_special_call(callinfo, klass, name, signature, KlassHandle(), false, CHECK); + LinkInfo link_info(klass, name, signature, KlassHandle(), /*check_access*/false); + LinkResolver::resolve_special_call(callinfo, link_info, CHECK); methodHandle method = callinfo.selected_method(); assert(method.not_null(), "should have thrown exception"); @@ -250,7 +251,8 @@ void JavaCalls::call_special(JavaValue* result, Handle receiver, KlassHandle kla void JavaCalls::call_static(JavaValue* result, KlassHandle klass, Symbol* name, Symbol* signature, JavaCallArguments* args, TRAPS) { CallInfo callinfo; - LinkResolver::resolve_static_call(callinfo, klass, name, signature, KlassHandle(), false, true, CHECK); + LinkInfo link_info(klass, name, signature, KlassHandle(), /*check_access*/false); + LinkResolver::resolve_static_call(callinfo, link_info, true, CHECK); methodHandle method = callinfo.selected_method(); assert(method.not_null(), "should have thrown exception"); diff --git a/hotspot/src/share/vm/runtime/reflection.cpp b/hotspot/src/share/vm/runtime/reflection.cpp index 85a2b0aade3..0e439e87137 100644 --- a/hotspot/src/share/vm/runtime/reflection.cpp +++ b/hotspot/src/share/vm/runtime/reflection.cpp @@ -831,9 +831,9 @@ methodHandle Reflection::resolve_interface_call(instanceKlassHandle klass, metho CallInfo info; Symbol* signature = method->signature(); Symbol* name = method->name(); - LinkResolver::resolve_interface_call(info, receiver, recv_klass, klass, - name, signature, - KlassHandle(), false, true, + LinkResolver::resolve_interface_call(info, receiver, recv_klass, + LinkInfo(klass, name, signature, KlassHandle(), false), + true, CHECK_(methodHandle())); return info.selected_method(); }