gems-kernel/source/THIRDPARTY/xnu/libkern/c++/OSCollectionIterator.cpp

332 lines
8.9 KiB
C++
Raw Normal View History

2024-06-03 16:29:39 +00:00
/*
* Copyright (c) 2000 Apple Computer, Inc. All rights reserved.
*
* @APPLE_OSREFERENCE_LICENSE_HEADER_START@
*
* This file contains Original Code and/or Modifications of Original Code
* as defined in and that are subject to the Apple Public Source License
* Version 2.0 (the 'License'). You may not use this file except in
* compliance with the License. The rights granted to you under the License
* may not be used to create, or enable the creation or redistribution of,
* unlawful or unlicensed copies of an Apple operating system, or to
* circumvent, violate, or enable the circumvention or violation of, any
* terms of an Apple operating system software license agreement.
*
* Please obtain a copy of the License at
* http://www.opensource.apple.com/apsl/ and read it before using this file.
*
* The Original Code and all software distributed under the License are
* distributed on an 'AS IS' basis, WITHOUT WARRANTY OF ANY KIND, EITHER
* EXPRESS OR IMPLIED, AND APPLE HEREBY DISCLAIMS ALL SUCH WARRANTIES,
* INCLUDING WITHOUT LIMITATION, ANY WARRANTIES OF MERCHANTABILITY,
* FITNESS FOR A PARTICULAR PURPOSE, QUIET ENJOYMENT OR NON-INFRINGEMENT.
* Please see the License for the specific language governing rights and
* limitations under the License.
*
* @APPLE_OSREFERENCE_LICENSE_HEADER_END@
*/
/* IOArray.h created by rsulack on Thu 11-Sep-1997 */
#define IOKIT_ENABLE_SHARED_PTR
#include <libkern/c++/OSArray.h>
#include <libkern/c++/OSCollection.h>
#include <libkern/c++/OSCollectionIterator.h>
#include <libkern/c++/OSLib.h>
#include <libkern/c++/OSSharedPtr.h>
#define super OSIterator
OSDefineMetaClassAndStructors(OSCollectionIterator, OSIterator)
bool
OSCollectionIterator::initWithCollection(const OSCollection *inColl)
{
if (!super::init() || !inColl) {
return false;
}
collection.reset(inColl, OSRetain);
collIterator = NULL;
initialUpdateStamp = 0;
valid = false;
return true;
}
OSSharedPtr<OSCollectionIterator>
OSCollectionIterator::withCollection(const OSCollection *inColl)
{
OSSharedPtr<OSCollectionIterator> me = OSMakeShared<OSCollectionIterator>();
if (me && !me->initWithCollection(inColl)) {
return nullptr;
}
return me;
}
void
OSCollectionIterator::free()
{
freeIteratorStorage();
collection.reset();
super::free();
}
void
OSCollectionIterator::reset()
{
valid = false;
bool initialized = initializeIteratorStorage();
if (!initialized) {
// reusing existing storage
void * storage = getIteratorStorage();
bzero(storage, collection->iteratorSize());
if (!collection->initIterator(storage)) {
return;
}
initialUpdateStamp = collection->updateStamp;
valid = true;
}
}
bool
OSCollectionIterator::isValid()
{
initializeIteratorStorage();
if (!valid || collection->updateStamp != initialUpdateStamp) {
return false;
}
return true;
}
bool
OSCollectionIterator::initializeIteratorStorage()
{
void * result = NULL;
bool initialized = false;
#if __LP64__
OSCollectionIteratorStorageType storageType = getStorageType();
switch (storageType) {
case OSCollectionIteratorStorageUnallocated:
if (collection->iteratorSize() > sizeof(inlineStorage) || isSubclassed()) {
collIterator = (void *)kalloc_data(collection->iteratorSize(), Z_WAITOK);
OSCONTAINER_ACCUMSIZE(collection->iteratorSize());
if (!collection->initIterator(collIterator)) {
kfree_data(collIterator, collection->iteratorSize());
OSCONTAINER_ACCUMSIZE(-((size_t) collection->iteratorSize()));
collIterator = NULL;
initialized = false;
setStorageType(OSCollectionIteratorStorageUnallocated);
} else {
setStorageType(OSCollectionIteratorStoragePointer);
result = collIterator;
initialized = true;
}
} else {
bzero(&inlineStorage[0], collection->iteratorSize());
if (!collection->initIterator(&inlineStorage[0])) {
bzero(&inlineStorage[0], collection->iteratorSize());
initialized = false;
setStorageType(OSCollectionIteratorStorageUnallocated);
} else {
setStorageType(OSCollectionIteratorStorageInline);
result = &inlineStorage[0];
initialized = true;
}
}
break;
case OSCollectionIteratorStoragePointer:
// already initialized
initialized = false;
break;
case OSCollectionIteratorStorageInline:
// already initialized
initialized = false;
break;
default:
panic("unexpected storage type %u", storageType);
}
#else
if (!collIterator) {
collIterator = (void *)kalloc_data(collection->iteratorSize(), Z_WAITOK);
OSCONTAINER_ACCUMSIZE(collection->iteratorSize());
if (!collection->initIterator(collIterator)) {
kfree_data(collIterator, collection->iteratorSize());
OSCONTAINER_ACCUMSIZE(-((size_t) collection->iteratorSize()));
collIterator = NULL;
initialized = false;
setStorageType(OSCollectionIteratorStorageUnallocated);
} else {
setStorageType(OSCollectionIteratorStoragePointer);
result = collIterator;
initialized = true;
}
}
#endif /* __LP64__ */
if (initialized) {
valid = true;
initialUpdateStamp = collection->updateStamp;
}
return initialized;
}
void *
OSCollectionIterator::getIteratorStorage()
{
void * result = NULL;
#if __LP64__
OSCollectionIteratorStorageType storageType = getStorageType();
switch (storageType) {
case OSCollectionIteratorStorageUnallocated:
result = NULL;
break;
case OSCollectionIteratorStoragePointer:
result = collIterator;
break;
case OSCollectionIteratorStorageInline:
result = &inlineStorage[0];
break;
default:
panic("unexpected storage type %u", storageType);
}
#else
OSCollectionIteratorStorageType storageType __assert_only = getStorageType();
assert(storageType == OSCollectionIteratorStoragePointer || storageType == OSCollectionIteratorStorageUnallocated);
result = collIterator;
#endif /* __LP64__ */
return result;
}
void
OSCollectionIterator::freeIteratorStorage()
{
#if __LP64__
OSCollectionIteratorStorageType storageType = getStorageType();
switch (storageType) {
case OSCollectionIteratorStorageUnallocated:
break;
case OSCollectionIteratorStoragePointer:
kfree_data(collIterator, collection->iteratorSize());
OSCONTAINER_ACCUMSIZE(-((size_t) collection->iteratorSize()));
collIterator = NULL;
setStorageType(OSCollectionIteratorStorageUnallocated);
break;
case OSCollectionIteratorStorageInline:
bzero(&inlineStorage[0], collection->iteratorSize());
setStorageType(OSCollectionIteratorStorageUnallocated);
break;
default:
panic("unexpected storage type %u", storageType);
}
#else
if (collIterator != NULL) {
assert(getStorageType() == OSCollectionIteratorStoragePointer);
kfree_data(collIterator, collection->iteratorSize());
OSCONTAINER_ACCUMSIZE(-((size_t) collection->iteratorSize()));
collIterator = NULL;
setStorageType(OSCollectionIteratorStorageUnallocated);
} else {
assert(getStorageType() == OSCollectionIteratorStorageUnallocated);
}
#endif /* __LP64__ */
}
bool
OSCollectionIterator::isSubclassed()
{
return getMetaClass() != OSCollectionIterator::metaClass;
}
OSCollectionIteratorStorageType
OSCollectionIterator::getStorageType()
{
#if __LP64__
// Storage type is in the most significant 2 bits of collIterator
return (OSCollectionIteratorStorageType)((uintptr_t)(collIterator) >> 62);
#else
if (collIterator != NULL) {
return OSCollectionIteratorStoragePointer;
} else {
return OSCollectionIteratorStorageUnallocated;
}
#endif /* __LP64__ */
}
void
OSCollectionIterator::setStorageType(OSCollectionIteratorStorageType storageType)
{
#if __LP64__
switch (storageType) {
case OSCollectionIteratorStorageUnallocated:
if (collIterator != NULL) {
assert(getStorageType() == OSCollectionIteratorStorageInline);
collIterator = NULL;
}
break;
case OSCollectionIteratorStoragePointer:
// Should already be set
assert(collIterator != NULL);
assert(getStorageType() == OSCollectionIteratorStoragePointer);
break;
case OSCollectionIteratorStorageInline:
// Set the two most sigificant bits of collIterator to 10b
collIterator = (void *)(((uintptr_t)collIterator & ~0xC000000000000000) | ((uintptr_t)OSCollectionIteratorStorageInline << 62));
break;
default:
panic("unexpected storage type %u", storageType);
}
#else
switch (storageType) {
case OSCollectionIteratorStorageUnallocated:
// Should already be set
assert(collIterator == NULL);
assert(getStorageType() == OSCollectionIteratorStorageUnallocated);
break;
case OSCollectionIteratorStoragePointer:
// Should already be set
assert(collIterator != NULL);
assert(getStorageType() == OSCollectionIteratorStoragePointer);
break;
case OSCollectionIteratorStorageInline:
panic("cannot use inline storage on LP32");
break;
default:
panic("unexpected storage type %u", storageType);
}
#endif /* __LP64__ */
}
OSObject *
OSCollectionIterator::getNextObject()
{
OSObject *retObj;
bool retVal;
void * storage;
if (!isValid()) {
return NULL;
}
storage = getIteratorStorage();
assert(storage != NULL);
retVal = collection->getNextObjectForIterator(storage, &retObj);
return (retVal)? retObj : NULL;
}