@ -39,83 +39,44 @@ You can contact Cyan Worlds, Inc. by email legal@cyan.com
Mead , WA 99021
Mead , WA 99021
* = = LICENSE = = */
* = = LICENSE = = */
# include "plRegistryKeyList.h"
# include "plRegistryHelpers.h"
# include "hsStream.h"
# include <algorithm>
# include <algorithm>
plRegistryKeyList : : plRegistryKeyList ( uint16_t classType )
# include "HeadSpin.h"
{
# include "hsStream.h"
fClassType = classType ;
fReffedStaticKeys = 0 ;
# include "pnKeyedObject/plKeyImp.h"
fLocked = 0 ;
# include "plRegistryHelpers.h"
fFlags = 0 ;
# include "plRegistryKeyList.h"
}
plRegistryKeyList : : ~ plRegistryKeyList ( )
plRegistryKeyList : : ~ plRegistryKeyList ( )
{
{
hsAssert ( fLocked = = 0 , " Key list still locked on delete " ) ;
std : : for_each ( fKeys . begin ( ) , fKeys . end ( ) ,
[ ] ( plKeyImp * key ) { if ( ! key - > ObjectIsLoaded ( ) ) delete key ; }
for ( int i = 0 ; i < fStaticKeys . size ( ) ; i + + )
) ;
{
plKeyImp * keyImp = fStaticKeys [ i ] ;
if ( ! keyImp - > ObjectIsLoaded ( ) )
delete keyImp ;
}
}
}
// Special dummy key that lets us set the return value of the GetName call.
plKeyImp * plRegistryKeyList : : FindKey ( const plString & keyName ) const
// Makes it easier to do STL searches.
class plSearchKeyImp : public plKeyImp
{
public :
plString fSearchKeyName ;
const plString & GetName ( ) const { return fSearchKeyName ; }
} ;
plKeyImp * plRegistryKeyList : : FindKey ( const plString & keyName )
{
{
static plSearchKeyImp searchKey ;
auto it = std : : find_if ( fKeys . begin ( ) , fKeys . end ( ) ,
searchKey . fSearchKeyName = keyName ;
[ & ] ( plKeyImp * key ) { return key - > GetName ( ) . CompareI ( keyName ) = = 0 ; }
) ;
// Search the static key list
if ( it ! = fKeys . end ( ) )
if ( fFlags & kStaticUnsorted )
{
// We're unsorted, brute force it. May do a separate search table in the
// future if this is a bottlneck
for ( int i = 0 ; i < fStaticKeys . size ( ) ; i + + )
{
plKeyImp * curKey = fStaticKeys [ i ] ;
if ( curKey & & ! keyName . Compare ( curKey - > GetName ( ) , plString : : kCaseInsensitive ) )
return curKey ;
}
}
else
{
// We're sorted, do a fast lookup
StaticVec : : const_iterator it = std : : lower_bound ( fStaticKeys . begin ( ) , fStaticKeys . end ( ) , & searchKey , KeySorter ( ) ) ;
if ( it ! = fStaticKeys . end ( ) & & ! keyName . Compare ( ( * it ) - > GetName ( ) , plString : : kCaseInsensitive ) )
return * it ;
return * it ;
}
else
return nullptr ;
// Search the dynamic key list
DynSet : : const_iterator dynIt = fDynamicKeys . find ( & searchKey ) ;
if ( dynIt ! = fDynamicKeys . end ( ) )
return * dynIt ;
return nil ;
}
}
plKeyImp * plRegistryKeyList : : FindKey ( const plUoid & uoid )
plKeyImp * plRegistryKeyList : : FindKey ( const plUoid & uoid ) const
{
{
uint32_t objectID = uoid . GetObjectID ( ) ;
uint32_t objectID = uoid . GetObjectID ( ) ;
// Key is dynamic or doesn't know it' s index. Do a find by name.
// Key is dynamic or doesn't know its index. Do a find by name.
if ( objectID = = 0 )
if ( objectID = = 0 )
return FindKey ( uoid . GetObjectName ( ) ) ;
return FindKey ( uoid . GetObjectName ( ) ) ;
// Direct lookup
// Direct lookup
if ( objectID < = fStatic Keys . size ( ) )
if ( objectID < = fKeys . size ( ) )
{
{
# ifdef PLASMA_EXTERNAL_RELEASE
# ifdef PLASMA_EXTERNAL_RELEASE
return fStaticKeys [ objectID - 1 ] ;
return fStaticKeys [ objectID - 1 ] ;
@ -123,8 +84,8 @@ plKeyImp* plRegistryKeyList::FindKey(const plUoid& uoid)
// If this is an internal release, our objectIDs might not match
// If this is an internal release, our objectIDs might not match
// because of local data. Verify that we have the right key by
// because of local data. Verify that we have the right key by
// name, and if it's wrong, do the slower find-by-name.
// name, and if it's wrong, do the slower find-by-name.
plKeyImp * keyImp = fStatic Keys [ objectID - 1 ] ;
plKeyImp * keyImp = fKeys [ objectID - 1 ] ;
if ( keyImp - > GetName ( ) . Compare ( uoid . GetObjectName ( ) , plString : : kCaseInsensitive ) ! = 0 )
if ( keyImp - > GetName ( ) . CompareI ( uoid . GetObjectName ( ) ) ! = 0 )
return FindKey ( uoid . GetObjectName ( ) ) ;
return FindKey ( uoid . GetObjectName ( ) ) ;
else
else
return keyImp ;
return keyImp ;
@ -135,29 +96,17 @@ plKeyImp* plRegistryKeyList::FindKey(const plUoid& uoid)
// because no one was using them. No worries. The resManager will catch this and
// because no one was using them. No worries. The resManager will catch this and
// reload our keys, then try again.
// reload our keys, then try again.
return nil ;
return nullptr ;
}
void plRegistryKeyList : : ILock ( )
{
fLocked + + ;
}
void plRegistryKeyList : : IUnlock ( )
{
fLocked - - ;
if ( fLocked = = 0 )
IRepack ( ) ;
}
}
bool plRegistryKeyList : : IterateKeys ( plRegistryKeyIterator * iterator )
bool plRegistryKeyList : : IterateKeys ( plRegistryKeyIterator * iterator )
{
{
ILock ( ) ;
ILock ( ) ;
for ( int i = 0 ; i < fStatic Keys. siz e( ) ; i + + )
for ( auto it = fKeys . begin ( ) ; it ! = fKeys . end ( ) ; + + it )
{
{
plKeyImp * keyImp = fStaticKeys [ i ] ;
plKeyImp * keyImp = * it ;
if ( keyImp ! = nil )
if ( keyImp )
{
{
if ( ! iterator - > EatKey ( plKey : : Make ( keyImp ) ) )
if ( ! iterator - > EatKey ( plKey : : Make ( keyImp ) ) )
{
{
@ -167,18 +116,6 @@ bool plRegistryKeyList::IterateKeys(plRegistryKeyIterator* iterator)
}
}
}
}
DynSet : : const_iterator it ;
for ( it = fDynamicKeys . begin ( ) ; it ! = fDynamicKeys . end ( ) ; it + + )
{
plKeyImp * keyImp = * it ;
hsAssert ( keyImp , " Shouldn't ever have a nil dynamic key " ) ;
if ( ! iterator - > EatKey ( plKey : : Make ( keyImp ) ) )
{
IUnlock ( ) ;
return false ;
}
}
IUnlock ( ) ;
IUnlock ( ) ;
return true ;
return true ;
}
}
@ -188,23 +125,30 @@ void plRegistryKeyList::AddKey(plKeyImp* key, LoadStatus& loadStatusChange)
loadStatusChange = kNoChange ;
loadStatusChange = kNoChange ;
hsAssert ( fLocked = = 0 , " Don't currently support adding keys while locked " ) ;
hsAssert ( fLocked = = 0 , " Don't currently support adding keys while locked " ) ;
if ( fLocked = = 0 & & key ! = nil )
if ( fLocked = = 0 & & key )
{
{
// If this is the first key added, we just became loaded
hsAssert ( std : : find ( fKeys . begin ( ) , fKeys . end ( ) , key ) = = fKeys . end ( ) , " Key already added " ) ;
if ( fDynamicKeys . empty ( ) )
loadStatusChange = kDynLoaded ;
hsAssert ( fDynamicKeys . find ( key ) = = fDynamicKeys . end ( ) , " Key already added " ) ;
// first key to be added?
fDynamicKeys . insert ( key ) ;
if ( fKeys . empty ( ) )
}
loadStatusChange = kTypeLoaded ;
}
void plRegistryKeyList : : SetKeyUsed ( plKeyImp * key )
// Objects that already have an object ID will be respected.
{
// Totally new keys will not have one, but keys from other sources (patches) will.
// If this is a static key, mark that we used it. Otherwise, just ignore it.
if ( key - > GetUoid ( ) . GetObjectID ( ) = = 0 )
{
fKeys . push_back ( key ) ;
key - > SetObjectID ( fKeys . size ( ) ) ;
}
else
{
uint32_t id = key - > GetUoid ( ) . GetObjectID ( ) ;
uint32_t id = key - > GetUoid ( ) . GetObjectID ( ) ;
if ( id > 0 )
if ( fKeys . size ( ) < id )
fReffedStaticKeys + + ;
fKeys . resize ( id ) ;
fKeys [ id - 1 ] = key ;
}
+ + fReffedKeys ;
}
}
}
bool plRegistryKeyList : : SetKeyUnused ( plKeyImp * key , LoadStatus & loadStatusChange )
bool plRegistryKeyList : : SetKeyUnused ( plKeyImp * key , LoadStatus & loadStatusChange )
@ -219,122 +163,50 @@ bool plRegistryKeyList::SetKeyUnused(plKeyImp* key, LoadStatus& loadStatusChange
return true ;
return true ;
}
}
// Check if it's a static key
uint32_t id = key - > GetUoid ( ) . GetObjectID ( ) ;
uint32_t id = key - > GetUoid ( ) . GetObjectID ( ) ;
hsAssert ( id < = fStaticKeys . size ( ) , " Bad static key id " ) ;
hsAssert ( id < = fKeys . size ( ) , " Bad static key id " ) ;
if ( id ! = 0 & & id < = fStaticKeys . size ( ) )
{
fReffedStaticKeys - - ;
if ( fLocked = = 0 )
IRepack ( ) ;
// That was our last used static key, we're static unloaded
// Fixed Keys will have id == 0. Let's just make sure we have it before we toss it.
if ( fReffedStaticKeys = = 0 )
if ( id = = 0 )
loadStatusChange = kStaticUnloaded ;
return true ;
}
// Try to find it in the dynamic key list
DynSet : : iterator dynIt = fDynamicKeys . find ( key ) ;
if ( dynIt ! = fDynamicKeys . end ( ) )
{
{
hsAssert ( fLocked = = 0 , " Don't currently support removing dynamic keys while locked " ) ;
hsAssert ( key - > GetUoid ( ) . GetLocation ( ) = = plLocation : : kGlobalFixedLoc , " key id == 0 but not fixed? " ) ;
if ( fLocked = = 0 )
if ( ! FindKey ( key - > GetName ( ) ) )
{
{
fDynamicKeys . erase ( dynIt ) ;
hsAssert ( false , " Couldn't find fixed key! " ) ;
delete key ;
// That was our last dynamic key, notify of dynamic unloaded
if ( fDynamicKeys . empty ( ) )
loadStatusChange = kDynUnloaded ;
return true ;
}
return false ;
}
hsAssert ( 0 , " Couldn't find this key, what is it? " ) ;
return false ;
return false ;
}
//// IRepack /////////////////////////////////////////////////////////////////
// Frees the memory for our static key array if none of them are loaded
void plRegistryKeyList : : IRepack ( )
{
if ( fReffedStaticKeys = = 0 & & ! fStaticKeys . empty ( ) )
{
for ( int i = 0 ; i < fStaticKeys . size ( ) ; i + + )
delete fStaticKeys [ i ] ;
fStaticKeys . clear ( ) ;
}
}
}
void plRegistryKeyList : : PrepForWrite ( )
{
// If we have any static keys already, we were read in. To keep from
// invalidating old key indexes any new keys have to go on the end, hence we're
// unsorted now.
if ( ! fStaticKeys . empty ( ) )
fFlags | = kStaticUnsorted ;
// If a dynamic keys doesn't have an object assigned to it, we're not writing
// it out. Figure out how many valid keys we have.
int numDynKeys = 0 ;
DynSet : : const_iterator cIt ;
for ( cIt = fDynamicKeys . begin ( ) ; cIt ! = fDynamicKeys . end ( ) ; cIt + + )
{
plKeyImp * key = * cIt ;
// We're only going to write out keys that have objects
if ( key - > ObjectIsLoaded ( ) )
numDynKeys + + ;
}
}
else if ( id > fKeys . size ( ) )
return false ;
// Start our new object id's after any already created ones
// Got that key, decrement the key counter
uint32_t objectID = fStaticKeys . size ( ) + 1 ;
- - fReffedKeys ;
// Make room for our new keys
if ( fReffedKeys = = 0 )
fStaticKeys . resize ( fStaticKeys . size ( ) + numDynKeys ) ;
loadStatusChange = kTypeUnloaded ;
return true ;
DynSet : : iterator it = fDynamicKeys . begin ( ) ;
while ( it ! = fDynamicKeys . end ( ) )
{
plKeyImp * key = * it ;
it + + ;
// If we're gonna use this key, tag it with it's object id and move it to the static array.
if ( key - > ObjectIsLoaded ( ) )
{
key - > SetObjectID ( objectID ) ;
fStaticKeys [ objectID - 1 ] = key ;
objectID + + ;
fReffedStaticKeys + + ;
fDynamicKeys . erase ( key ) ;
}
}
}
}
void plRegistryKeyList : : Read ( hsStream * s )
void plRegistryKeyList : : Read ( hsStream * s )
{
{
uint32_t keyListLen = s - > ReadLE32 ( ) ;
uint32_t keyListLen = s - > ReadLE32 ( ) ;
if ( ! fStatic Keys . empty ( ) )
if ( ! fKeys . empty ( ) )
{
{
s - > Skip ( keyListLen ) ;
s - > Skip ( keyListLen ) ;
return ;
return ;
}
}
fFlags = s - > ReadByte ( ) ;
// deprecated flags. used to indicate alphabetically sorted keys for some "optimization"
// that really appeared to do nothing. no loss.
s - > ReadByte ( ) ;
uint32_t numKeys = s - > ReadLE32 ( ) ;
uint32_t numKeys = s - > ReadLE32 ( ) ;
fStatic Keys . resiz e ( numKeys ) ;
fKeys . reserve ( numKeys ) ;
for ( int i = 0 ; i < numKeys ; i + + )
for ( u int32_ t i = 0 ; i < numKeys ; + + i )
{
{
plKeyImp * newKey = new plKeyImp ;
plKeyImp * newKey = new plKeyImp ;
newKey - > Read ( s ) ;
newKey - > Read ( s ) ;
fStaticKeys [ i ] = newKey ;
fKeys . push_back ( newKey ) ;
}
}
}
}
@ -343,17 +215,13 @@ void plRegistryKeyList::Write(hsStream* s)
// Save space for the length of our data
// Save space for the length of our data
uint32_t beginPos = s - > GetPosition ( ) ;
uint32_t beginPos = s - > GetPosition ( ) ;
s - > WriteLE32 ( 0 ) ;
s - > WriteLE32 ( 0 ) ;
s - > WriteByte ( fFlags ) ;
s - > WriteByte ( 0 ) ; // Deprecated flags
int numKeys = fStaticKeys . size ( ) ;
s - > WriteLE32 ( fKeys . size ( ) ) ;
s - > WriteLE32 ( numKeys ) ;
// Write out all our keys (anything in dynamic is unused, so just ignore those)
// Write out all our keys
for ( int i = 0 ; i < numKeys ; i + + )
for ( auto it = fKeys . begin ( ) ; it ! = fKeys . end ( ) ; + + it )
{
( * it ) - > Write ( s ) ;
plKeyImp * key = fStaticKeys [ i ] ;
key - > Write ( s ) ;
}
// Go back to the start and write the length of our data
// Go back to the start and write the length of our data
uint32_t endPos = s - > GetPosition ( ) ;
uint32_t endPos = s - > GetPosition ( ) ;