8310874: Runthese30m crashes with klass should be in the placeholders during verification

Reviewed-by: dholmes, iklam
This commit is contained in:
Coleen Phillimore 2023-09-21 12:17:34 +00:00
parent 6a4b665550
commit b3d75fe12e
3 changed files with 25 additions and 14 deletions

View File

@ -27,15 +27,11 @@
#include "classfile/classLoaderDataGraph.hpp" #include "classfile/classLoaderDataGraph.hpp"
#include "classfile/dictionary.hpp" #include "classfile/dictionary.hpp"
#include "classfile/loaderConstraints.hpp" #include "classfile/loaderConstraints.hpp"
#include "classfile/placeholders.hpp"
#include "logging/log.hpp" #include "logging/log.hpp"
#include "memory/resourceArea.hpp" #include "memory/resourceArea.hpp"
#include "oops/klass.inline.hpp" #include "oops/klass.inline.hpp"
#include "oops/oop.inline.hpp"
#include "oops/symbolHandle.hpp" #include "oops/symbolHandle.hpp"
#include "runtime/handles.inline.hpp"
#include "runtime/mutexLocker.hpp" #include "runtime/mutexLocker.hpp"
#include "runtime/safepoint.hpp"
#include "utilities/resourceHash.hpp" #include "utilities/resourceHash.hpp"
// Overview // Overview
@ -447,6 +443,23 @@ InstanceKlass* LoaderConstraintTable::find_constrained_klass(Symbol* name,
return nullptr; return nullptr;
} }
// Removes a class that was added to the table then class loading subsequently failed for this class,
// so we don't have a dangling pointer to InstanceKlass in the LoaderConstraintTable.
void LoaderConstraintTable::remove_failed_loaded_klass(InstanceKlass* klass,
ClassLoaderData* loader) {
Symbol* name = klass->name();
LoaderConstraint *p = find_loader_constraint(name, loader);
if (p != nullptr && p->klass() != nullptr && p->klass() == klass) {
// If this is the klass in the constraint, the error was OOM from the ClassLoader.addClass() call.
// Other errors during loading (eg. constraint violations) will not have added this klass.
log_info(class, loader, constraints)("removing klass %s: failed to load", name->as_C_string());
// We only null out the class, since the constraint for the class name for this loader is still valid as
// it was added when checking signature loaders for a method or field resolution.
p->set_klass(nullptr);
}
}
void LoaderConstraintTable::merge_loader_constraints(Symbol* class_name, void LoaderConstraintTable::merge_loader_constraints(Symbol* class_name,
LoaderConstraint* p1, LoaderConstraint* p1,
LoaderConstraint* p2, LoaderConstraint* p2,
@ -511,15 +524,8 @@ void LoaderConstraintTable::verify() {
// We found the class in the dictionary, so we should // We found the class in the dictionary, so we should
// make sure that the Klass* matches what we already have. // make sure that the Klass* matches what we already have.
guarantee(k == probe->klass(), "klass should be in dictionary"); guarantee(k == probe->klass(), "klass should be in dictionary");
} else { // If we don't find the class in the dictionary, it is
// If we don't find the class in the dictionary, it // in the process of loading and may or may not be in the placeholder table.
// has to be in the placeholders table.
PlaceholderEntry* entry = PlaceholderTable::get_entry(name, loader_data);
// The InstanceKlass might not be on the entry, so the only
// thing we can check here is whether we were successful in
// finding the class in the placeholders table.
guarantee(entry != nullptr, "klass should be in the placeholders");
} }
} }
for (int n = 0; n< probe->num_loaders(); n++) { for (int n = 0; n< probe->num_loaders(); n++) {

View File

@ -1,5 +1,5 @@
/* /*
* Copyright (c) 2003, 2022, Oracle and/or its affiliates. All rights reserved. * Copyright (c) 2003, 2023, Oracle and/or its affiliates. All rights reserved.
* DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER. * DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER.
* *
* This code is free software; you can redistribute it and/or modify it * This code is free software; you can redistribute it and/or modify it
@ -59,6 +59,7 @@ public:
// Class loader constraints // Class loader constraints
static bool check_or_update(InstanceKlass* k, ClassLoaderData* loader, Symbol* name); static bool check_or_update(InstanceKlass* k, ClassLoaderData* loader, Symbol* name);
static void remove_failed_loaded_klass(InstanceKlass* k, ClassLoaderData* loader);
static void purge_loader_constraints(); static void purge_loader_constraints();
static void print_table_statistics(outputStream* st); static void print_table_statistics(outputStream* st);

View File

@ -1526,6 +1526,10 @@ InstanceKlass* SystemDictionary::find_or_define_instance_class(Symbol* class_nam
} else if (HAS_PENDING_EXCEPTION) { } else if (HAS_PENDING_EXCEPTION) {
assert(defined_k == nullptr, "Should not have a klass if there's an exception"); assert(defined_k == nullptr, "Should not have a klass if there's an exception");
k->class_loader_data()->add_to_deallocate_list(k); k->class_loader_data()->add_to_deallocate_list(k);
// Also remove this InstanceKlass from the LoaderConstraintTable if added.
MutexLocker ml(SystemDictionary_lock);
LoaderConstraintTable::remove_failed_loaded_klass(k, class_loader_data(class_loader));
} }
return defined_k; return defined_k;
} }