Skip to content

Commit 15dd8f3

Browse files
committedJan 13, 2021
8259275: JRuby crashes while resolving invokedynamic instruction
Reviewed-by: iklam, minqi, lfoltan
1 parent 1cf2378 commit 15dd8f3

File tree

8 files changed

+196
-38
lines changed

8 files changed

+196
-38
lines changed
 

‎src/hotspot/share/classfile/classListParser.cpp

+25-23
Original file line numberDiff line numberDiff line change
@@ -1,5 +1,5 @@
11
/*
2-
* Copyright (c) 2015, 2020, Oracle and/or its affiliates. All rights reserved.
2+
* Copyright (c) 2015, 2021, Oracle and/or its affiliates. All rights reserved.
33
* DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER.
44
*
55
* This code is free software; you can redistribute it and/or modify it
@@ -460,19 +460,34 @@ bool ClassListParser::is_matching_cp_entry(constantPoolHandle &pool, int cp_inde
460460
}
461461
return true;
462462
}
463-
464463
void ClassListParser::resolve_indy(Symbol* class_name_symbol, TRAPS) {
464+
ClassListParser::resolve_indy_impl(class_name_symbol, THREAD);
465+
if (HAS_PENDING_EXCEPTION) {
466+
ResourceMark rm(THREAD);
467+
char* ex_msg = (char*)"";
468+
oop message = java_lang_Throwable::message(PENDING_EXCEPTION);
469+
if (message != NULL) {
470+
ex_msg = java_lang_String::as_utf8_string(message);
471+
}
472+
log_warning(cds)("resolve_indy for class %s has encountered exception: %s %s",
473+
class_name_symbol->as_C_string(),
474+
PENDING_EXCEPTION->klass()->external_name(),
475+
ex_msg);
476+
CLEAR_PENDING_EXCEPTION;
477+
}
478+
}
479+
480+
void ClassListParser::resolve_indy_impl(Symbol* class_name_symbol, TRAPS) {
465481
Handle class_loader(THREAD, SystemDictionary::java_system_loader());
466482
Handle protection_domain;
467-
Klass* klass = SystemDictionary::resolve_or_fail(class_name_symbol, class_loader, protection_domain, true, THREAD); // FIXME should really be just a lookup
483+
Klass* klass = SystemDictionary::resolve_or_fail(class_name_symbol, class_loader, protection_domain, true, CHECK); // FIXME should really be just a lookup
468484
if (klass != NULL && klass->is_instance_klass()) {
469485
InstanceKlass* ik = InstanceKlass::cast(klass);
470486
if (SystemDictionaryShared::has_class_failed_verification(ik)) {
471487
// don't attempt to resolve indy on classes that has previously failed verification
472488
return;
473489
}
474-
MetaspaceShared::try_link_class(ik, THREAD);
475-
assert(!HAS_PENDING_EXCEPTION, "unexpected exception");
490+
MetaspaceShared::try_link_class(ik, CHECK);
476491

477492
ConstantPool* cp = ik->constants();
478493
ConstantPoolCache* cpcache = cp->cache();
@@ -484,36 +499,23 @@ void ClassListParser::resolve_indy(Symbol* class_name_symbol, TRAPS) {
484499
constantPoolHandle pool(THREAD, cp);
485500
if (pool->tag_at(pool_index).is_invoke_dynamic()) {
486501
BootstrapInfo bootstrap_specifier(pool, pool_index, indy_index);
487-
Handle bsm = bootstrap_specifier.resolve_bsm(THREAD);
502+
Handle bsm = bootstrap_specifier.resolve_bsm(CHECK);
488503
if (!SystemDictionaryShared::is_supported_invokedynamic(&bootstrap_specifier)) {
489504
log_debug(cds, lambda)("is_supported_invokedynamic check failed for cp_index %d", pool_index);
490505
continue;
491506
}
492-
if (is_matching_cp_entry(pool, pool_index, THREAD)) {
507+
bool matched = is_matching_cp_entry(pool, pool_index, CHECK);
508+
if (matched) {
493509
found = true;
494510
CallInfo info;
495-
bool is_done = bootstrap_specifier.resolve_previously_linked_invokedynamic(info, THREAD);
511+
bool is_done = bootstrap_specifier.resolve_previously_linked_invokedynamic(info, CHECK);
496512
if (!is_done) {
497513
// resolve it
498514
Handle recv;
499-
LinkResolver::resolve_invoke(info, recv, pool, indy_index, Bytecodes::_invokedynamic, THREAD);
515+
LinkResolver::resolve_invoke(info, recv, pool, indy_index, Bytecodes::_invokedynamic, CHECK);
500516
break;
501517
}
502518
cpce->set_dynamic_call(pool, info);
503-
if (HAS_PENDING_EXCEPTION) {
504-
ResourceMark rm(THREAD);
505-
tty->print("resolve_indy for class %s has", class_name_symbol->as_C_string());
506-
oop message = java_lang_Throwable::message(PENDING_EXCEPTION);
507-
if (message != NULL) {
508-
char* ex_msg = java_lang_String::as_utf8_string(message);
509-
tty->print_cr(" exception pending '%s %s'",
510-
PENDING_EXCEPTION->klass()->external_name(), ex_msg);
511-
} else {
512-
tty->print_cr(" exception pending %s ",
513-
PENDING_EXCEPTION->klass()->external_name());
514-
}
515-
exit(1);
516-
}
517519
}
518520
}
519521
}

‎src/hotspot/share/classfile/classListParser.hpp

+2-1
Original file line numberDiff line numberDiff line change
@@ -1,5 +1,5 @@
11
/*
2-
* Copyright (c) 2015, 2019, Oracle and/or its affiliates. All rights reserved.
2+
* Copyright (c) 2015, 2021, Oracle and/or its affiliates. All rights reserved.
33
* DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER.
44
*
55
* This code is free software; you can redistribute it and/or modify it
@@ -114,6 +114,7 @@ class ClassListParser : public StackObj {
114114
bool is_matching_cp_entry(constantPoolHandle &pool, int cp_index, TRAPS);
115115

116116
void resolve_indy(Symbol* class_name_symbol, TRAPS);
117+
void resolve_indy_impl(Symbol* class_name_symbol, TRAPS);
117118
public:
118119
ClassListParser(const char* file);
119120
~ClassListParser();

‎src/hotspot/share/oops/constantPool.cpp

+9-10
Original file line numberDiff line numberDiff line change
@@ -1,5 +1,5 @@
11
/*
2-
* Copyright (c) 1997, 2020, Oracle and/or its affiliates. All rights reserved.
2+
* Copyright (c) 1997, 2021, Oracle and/or its affiliates. All rights reserved.
33
* DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER.
44
*
55
* This code is free software; you can redistribute it and/or modify it
@@ -406,15 +406,14 @@ void ConstantPool::remove_unshareable_info() {
406406
_flags |= (_on_stack | _is_shared);
407407
int num_klasses = 0;
408408
for (int index = 1; index < length(); index++) { // Index 0 is unused
409-
if (!DynamicDumpSharedSpaces) {
410-
assert(!tag_at(index).is_unresolved_klass_in_error(), "This must not happen during static dump time");
411-
} else {
412-
if (tag_at(index).is_unresolved_klass_in_error() ||
413-
tag_at(index).is_method_handle_in_error() ||
414-
tag_at(index).is_method_type_in_error() ||
415-
tag_at(index).is_dynamic_constant_in_error()) {
416-
tag_at_put(index, JVM_CONSTANT_UnresolvedClass);
417-
}
409+
if (tag_at(index).is_unresolved_klass_in_error()) {
410+
tag_at_put(index, JVM_CONSTANT_UnresolvedClass);
411+
} else if (tag_at(index).is_method_handle_in_error()) {
412+
tag_at_put(index, JVM_CONSTANT_MethodHandle);
413+
} else if (tag_at(index).is_method_type_in_error()) {
414+
tag_at_put(index, JVM_CONSTANT_MethodType);
415+
} else if (tag_at(index).is_dynamic_constant_in_error()) {
416+
tag_at_put(index, JVM_CONSTANT_Dynamic);
418417
}
419418
if (tag_at(index).is_klass()) {
420419
// This class was resolved as a side effect of executing Java code

‎test/hotspot/jtreg/TEST.groups

+2-1
Original file line numberDiff line numberDiff line change
@@ -1,5 +1,5 @@
11
#
2-
# Copyright (c) 2013, 2020, Oracle and/or its affiliates. All rights reserved.
2+
# Copyright (c) 2013, 2021, Oracle and/or its affiliates. All rights reserved.
33
# DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER.
44
#
55
# This code is free software; you can redistribute it and/or modify it
@@ -336,6 +336,7 @@ hotspot_appcds_dynamic = \
336336
-runtime/cds/appcds/LambdaEagerInit.java \
337337
-runtime/cds/appcds/LambdaProxyClasslist.java \
338338
-runtime/cds/appcds/LambdaVerificationFailedDuringDump.java \
339+
-runtime/cds/appcds/LambdaWithOldClass.java \
339340
-runtime/cds/appcds/LongClassListPath.java \
340341
-runtime/cds/appcds/LotsOfClasses.java \
341342
-runtime/cds/appcds/NonExistClasspath.java \

‎test/hotspot/jtreg/runtime/cds/appcds/BadBSM.java

+3-3
Original file line numberDiff line numberDiff line change
@@ -1,5 +1,5 @@
11
/*
2-
* Copyright (c) 2020, Oracle and/or its affiliates. All rights reserved.
2+
* Copyright (c) 2020, 2021, Oracle and/or its affiliates. All rights reserved.
33
* DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER.
44
*
55
* This code is free software; you can redistribute it and/or modify it
@@ -44,7 +44,7 @@ public static void main(String[] args) throws Exception {
4444
TestCommon.list("WrongBSM",
4545
"@lambda-proxy WrongBSM 7"),
4646
"-Xlog:cds+lambda=debug");
47-
out.shouldHaveExitValue(0);
48-
out.shouldContain( "is_supported_invokedynamic check failed for cp_index 7");
47+
out.shouldHaveExitValue(0)
48+
.shouldContain("resolve_indy for class WrongBSM has encountered exception: java.lang.NoSuchMethodError");
4949
}
5050
}
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,80 @@
1+
/*
2+
* Copyright (c) 2021, Oracle and/or its affiliates. All rights reserved.
3+
* DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER.
4+
*
5+
* This code is free software; you can redistribute it and/or modify it
6+
* under the terms of the GNU General Public License version 2 only, as
7+
* published by the Free Software Foundation.
8+
*
9+
* This code is distributed in the hope that it will be useful, but WITHOUT
10+
* ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or
11+
* FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License
12+
* version 2 for more details (a copy is included in the LICENSE file that
13+
* accompanied this code).
14+
*
15+
* You should have received a copy of the GNU General Public License version
16+
* 2 along with this work; if not, write to the Free Software Foundation,
17+
* Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA.
18+
*
19+
* Please contact Oracle, 500 Oracle Parkway, Redwood Shores, CA 94065 USA
20+
* or visit www.oracle.com if you need additional information or have any
21+
* questions.
22+
*
23+
*/
24+
25+
/*
26+
* @test
27+
* @bug 8259275
28+
* @summary VM should not crash during CDS dump and run time when a lambda
29+
* expression references an old version of class.
30+
* @requires vm.cds
31+
* @library /test/lib
32+
* @compile test-classes/OldClass.jasm
33+
* @compile test-classes/LambdaWithOldClassApp.java
34+
* @run driver LambdaWithOldClass
35+
*/
36+
37+
import jdk.test.lib.cds.CDSOptions;
38+
import jdk.test.lib.cds.CDSTestUtils;
39+
import jdk.test.lib.process.OutputAnalyzer;
40+
import jdk.test.lib.process.ProcessTools;
41+
42+
public class LambdaWithOldClass {
43+
44+
public static void main(String[] args) throws Exception {
45+
String mainClass = "LambdaWithOldClassApp";
46+
String namePrefix = "lambdawitholdclass";
47+
JarBuilder.build(namePrefix, mainClass, "OldClass", "TestInterface");
48+
49+
String appJar = TestCommon.getTestJar(namePrefix + ".jar");
50+
String classList = namePrefix + ".list";
51+
String archiveName = namePrefix + ".jsa";
52+
53+
// dump class list
54+
ProcessBuilder pb = ProcessTools.createTestJvm(
55+
"-XX:DumpLoadedClassList=" + classList,
56+
"-cp", appJar,
57+
mainClass);
58+
OutputAnalyzer output = TestCommon.executeAndLog(pb, namePrefix);
59+
output.shouldHaveExitValue(0);
60+
61+
// create archive with the class list
62+
CDSOptions opts = (new CDSOptions())
63+
.addPrefix("-XX:ExtraSharedClassListFile=" + classList,
64+
"-cp", appJar,
65+
"-Xlog:class+load,cds")
66+
.setArchiveName(archiveName);
67+
CDSTestUtils.createArchiveAndCheck(opts);
68+
69+
// run with archive
70+
CDSOptions runOpts = (new CDSOptions())
71+
.addPrefix("-cp", appJar, "-Xlog:class+load,cds=debug")
72+
.setArchiveName(archiveName)
73+
.setUseVersion(false)
74+
.addSuffix(mainClass);
75+
output = CDSTestUtils.runWithArchive(runOpts);
76+
output.shouldContain("[class,load] LambdaWithOldClassApp source: shared objects file")
77+
.shouldMatch(".class.load. LambdaWithOldClassApp[$][$]Lambda[$].*/0x.*source:.*LambdaWithOldClassApp")
78+
.shouldHaveExitValue(0);
79+
}
80+
}
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,38 @@
1+
/*
2+
* Copyright (c) 2021, Oracle and/or its affiliates. All rights reserved.
3+
* DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER.
4+
*
5+
* This code is free software; you can redistribute it and/or modify it
6+
* under the terms of the GNU General Public License version 2 only, as
7+
* published by the Free Software Foundation.
8+
*
9+
* This code is distributed in the hope that it will be useful, but WITHOUT
10+
* ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or
11+
* FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License
12+
* version 2 for more details (a copy is included in the LICENSE file that
13+
* accompanied this code).
14+
*
15+
* You should have received a copy of the GNU General Public License version
16+
* 2 along with this work; if not, write to the Free Software Foundation,
17+
* Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA.
18+
*
19+
* Please contact Oracle, 500 Oracle Parkway, Redwood Shores, CA 94065 USA
20+
* or visit www.oracle.com if you need additional information or have any
21+
* questions.
22+
*
23+
*/
24+
interface TestInterface {
25+
public void apply(OldClass c);
26+
}
27+
28+
public class LambdaWithOldClassApp {
29+
public static void main(String args[]) {
30+
doit((c) -> {
31+
System.out.println("c = " + c);
32+
});
33+
}
34+
static void doit(TestInterface i) {
35+
OldClass c = new OldClass();
36+
i.apply(c);
37+
}
38+
}
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,37 @@
1+
/*
2+
* Copyright (c) 2021, Oracle and/or its affiliates. All rights reserved.
3+
* DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER.
4+
*
5+
* This code is free software; you can redistribute it and/or modify it
6+
* under the terms of the GNU General Public License version 2 only, as
7+
* published by the Free Software Foundation.
8+
*
9+
* This code is distributed in the hope that it will be useful, but WITHOUT
10+
* ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or
11+
* FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License
12+
* version 2 for more details (a copy is included in the LICENSE file that
13+
* accompanied this code).
14+
*
15+
* You should have received a copy of the GNU General Public License version
16+
* 2 along with this work; if not, write to the Free Software Foundation,
17+
* Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA.
18+
*
19+
* Please contact Oracle, 500 Oracle Parkway, Redwood Shores, CA 94065 USA
20+
* or visit www.oracle.com if you need additional information or have any
21+
* questions.
22+
*
23+
*/
24+
super class OldClass
25+
version 49:0
26+
{
27+
28+
29+
Method "<init>":"()V"
30+
stack 1 locals 1
31+
{
32+
aload_0;
33+
invokespecial Method java/lang/Object."<init>":"()V";
34+
return;
35+
}
36+
37+
} // end Class OldClass

0 commit comments

Comments
 (0)
Please sign in to comment.