8310874: Runthese30m crashes with klass should be in the placeholders during verification
Reviewed-by: dholmes, iklam
This commit is contained in:
parent
6a4b665550
commit
b3d75fe12e
@ -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++) {
|
||||||
|
@ -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);
|
||||||
|
@ -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;
|
||||||
}
|
}
|
||||||
|
Loading…
x
Reference in New Issue
Block a user