/* * Copyright (c) 2020, 2024, Oracle and/or its affiliates. All rights reserved. * Copyright (c) 2020 SAP SE. 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 * under the terms of the GNU General Public License version 2 only, as * published by the Free Software Foundation. * * This code is distributed in the hope that it will be useful, but WITHOUT * ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or * FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License * version 2 for more details (a copy is included in the LICENSE file that * accompanied this code). * * You should have received a copy of the GNU General Public License version * 2 along with this work; if not, write to the Free Software Foundation, * Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA. * * Please contact Oracle, 500 Oracle Parkway, Redwood Shores, CA 94065 USA * or visit www.oracle.com if you need additional information or have any * questions. * */ #include "precompiled.hpp" #include "memory/allocation.inline.hpp" #include "memory/classLoaderMetaspace.hpp" #include "memory/metaspace.hpp" #include "memory/metaspace/virtualSpaceList.hpp" #include "runtime/mutex.hpp" #include "runtime/mutexLocker.hpp" #include "runtime/os.hpp" #include "unittest.hpp" using namespace metaspace; // Test the cheerful multitude of metaspace-contains-functions. class MetaspaceIsMetaspaceObjTest { Mutex* _lock; ClassLoaderMetaspace* _ms; public: MetaspaceIsMetaspaceObjTest() : _lock(nullptr), _ms(nullptr) {} ~MetaspaceIsMetaspaceObjTest() { delete _ms; delete _lock; } void do_test(Metaspace::MetadataType mdType) { _lock = new Mutex(Monitor::nosafepoint, "gtest-IsMetaspaceObjTest_lock"); { MutexLocker ml(_lock, Mutex::_no_safepoint_check_flag); _ms = new ClassLoaderMetaspace(_lock, Metaspace::StandardMetaspaceType); } const MetaspaceObj* p = (MetaspaceObj*) _ms->allocate(42, mdType); // Test MetaspaceObj::is_metaspace_object ASSERT_TRUE(MetaspaceObj::is_valid(p)); // A misaligned object shall not be recognized const MetaspaceObj* p_misaligned = (MetaspaceObj*)((address)p) + 1; ASSERT_FALSE(MetaspaceObj::is_valid(p_misaligned)); // Test VirtualSpaceList::contains const VirtualSpaceList* const vslist = (mdType == Metaspace::ClassType && Metaspace::using_class_space()) ? VirtualSpaceList::vslist_class() : VirtualSpaceList::vslist_nonclass(); ASSERT_TRUE(vslist->contains((MetaWord*)p)); // A misaligned pointer shall still be recognized by list::contains ASSERT_TRUE(vslist->contains((MetaWord*)((address)p) + 1)); // Now for some bogus values ASSERT_FALSE(MetaspaceObj::is_valid((MetaspaceObj*)nullptr)); // Should exercise various paths in MetaspaceObj::is_valid() ASSERT_FALSE(MetaspaceObj::is_valid((MetaspaceObj*)1024)); ASSERT_FALSE(MetaspaceObj::is_valid((MetaspaceObj*)8192)); MetaspaceObj* p_stack = (MetaspaceObj*) &_lock; ASSERT_FALSE(MetaspaceObj::is_valid(p_stack)); MetaspaceObj* p_heap = (MetaspaceObj*) os::malloc(41, mtInternal); ASSERT_FALSE(MetaspaceObj::is_valid(p_heap)); os::free(p_heap); // Test Metaspace::contains_xxx ASSERT_TRUE(Metaspace::contains(p)); ASSERT_TRUE(Metaspace::contains_non_shared(p)); delete _ms; _ms = nullptr; delete _lock; _lock = nullptr; } }; TEST_VM(metaspace, is_metaspace_obj_non_class) { MetaspaceIsMetaspaceObjTest test; test.do_test(Metaspace::NonClassType); } TEST_VM(metaspace, is_metaspace_obj_class) { MetaspaceIsMetaspaceObjTest test; test.do_test(Metaspace::ClassType); }