You can not select more than 25 topics
Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
357 lines
9.6 KiB
357 lines
9.6 KiB
/*==LICENSE==* |
|
|
|
CyanWorlds.com Engine - MMOG client, server and tools |
|
Copyright (C) 2011 Cyan Worlds, Inc. |
|
|
|
This program is free software: you can redistribute it and/or modify |
|
it under the terms of the GNU General Public License as published by |
|
the Free Software Foundation, either version 3 of the License, or |
|
(at your option) any later version. |
|
|
|
This program 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 for more details. |
|
|
|
You should have received a copy of the GNU General Public License |
|
along with this program. If not, see <http://www.gnu.org/licenses/>. |
|
|
|
Additional permissions under GNU GPL version 3 section 7 |
|
|
|
If you modify this Program, or any covered work, by linking or |
|
combining it with any of RAD Game Tools Bink SDK, Autodesk 3ds Max SDK, |
|
NVIDIA PhysX SDK, Microsoft DirectX SDK, OpenSSL library, Independent |
|
JPEG Group JPEG library, Microsoft Windows Media SDK, or Apple QuickTime SDK |
|
(or a modified version of those libraries), |
|
containing parts covered by the terms of the Bink SDK EULA, 3ds Max EULA, |
|
PhysX SDK EULA, DirectX SDK EULA, OpenSSL and SSLeay licenses, IJG |
|
JPEG Library README, Windows Media SDK EULA, or QuickTime SDK EULA, the |
|
licensors of this Program grant you additional |
|
permission to convey the resulting work. Corresponding Source for a |
|
non-source form of such a combination shall include the source code for |
|
the parts of OpenSSL and IJG JPEG Library used as well as that of the covered |
|
work. |
|
|
|
You can contact Cyan Worlds, Inc. by email legal@cyan.com |
|
or by snail mail at: |
|
Cyan Worlds, Inc. |
|
14617 N Newport Hwy |
|
Mead, WA 99021 |
|
|
|
*==LICENSE==*/ |
|
|
|
#define PLFACTORY_PRIVATE |
|
#include "hsTypes.h" |
|
#include "plFactory.h" |
|
#include "hsStream.h" |
|
#include "plCreatable.h" |
|
#include "plCreator.h" |
|
#include "hsUtils.h" |
|
|
|
// For class names |
|
#include "plCreatableStrings.h" |
|
|
|
|
|
static plFactory* theFactory = nil; |
|
|
|
plFactory::plFactory() |
|
{ |
|
fCreators.SetCountAndZero(plCreatableIndex::plNumClassIndices); |
|
} |
|
|
|
plFactory::~plFactory() |
|
{ |
|
} |
|
|
|
hsBool plFactory::ICreateTheFactory() |
|
{ |
|
if( theFactory ) |
|
return true; |
|
|
|
theFactory = TRACKED_NEW plFactory; |
|
|
|
return theFactory != nil; |
|
} |
|
|
|
uint16_t plFactory::IGetNumClasses() |
|
{ |
|
return plCreatableIndex::plNumClassIndices; |
|
} |
|
|
|
void plFactory::IForceShutdown() |
|
{ |
|
int i; |
|
for( i = 0; i < fCreators.GetCount(); i++ ) |
|
{ |
|
if( fCreators[i] ) |
|
{ |
|
hsRefCnt_SafeUnRef(this); |
|
fCreators[i] = nil; |
|
} |
|
} |
|
} |
|
|
|
void plFactory::IShutdown() |
|
{ |
|
delete theFactory; |
|
theFactory = nil; |
|
} |
|
|
|
uint16_t plFactory::IRegister(uint16_t hClass, plCreator* worker) |
|
{ |
|
delete fCreators[hClass]; |
|
fCreators[hClass] = worker; |
|
return hClass; |
|
} |
|
|
|
// |
|
// return true if creator exists |
|
// |
|
bool plFactory::CanCreate(uint16_t hClass) |
|
{ |
|
if( hClass & 0x8000 ) // nil creatable |
|
return false; |
|
|
|
if( !theFactory && !ICreateTheFactory() ) // no factory |
|
return false; |
|
|
|
if( hClass >= theFactory->IGetNumClasses() ) // invalid index |
|
return false; |
|
|
|
return ( theFactory->fCreators[ hClass ] != nil ); // check creator |
|
} |
|
|
|
plCreatable* plFactory::ICreate(uint16_t hClass) |
|
{ |
|
if (CanCreate(hClass)) |
|
{ |
|
return fCreators[hClass]->Create(); |
|
} |
|
|
|
if (!(hClass & 0x8000)) |
|
{ |
|
hsAssert( false, "Invalid class index or nil creator : plFactory::Create()" ); |
|
} |
|
return nil; |
|
} |
|
|
|
void plFactory::UnRegister(uint16_t hClass, plCreator* worker) |
|
{ |
|
if( theFactory ) |
|
{ |
|
theFactory->IUnRegister(hClass); |
|
hsRefCnt_SafeUnRef(theFactory); |
|
if( theFactory->RefCnt() < 2 ) |
|
IShutdown(); |
|
} |
|
} |
|
|
|
void plFactory::IUnRegister(uint16_t hClass) |
|
{ |
|
fCreators[hClass] = nil; |
|
} |
|
|
|
uint16_t plFactory::Register(uint16_t hClass, plCreator* worker) |
|
{ |
|
if( !theFactory && !ICreateTheFactory() ) |
|
return nil; |
|
|
|
hsRefCnt_SafeRef(theFactory); |
|
return theFactory->IRegister(hClass, worker); |
|
} |
|
|
|
plCreatable* plFactory::Create(uint16_t hClass) |
|
{ |
|
if( !theFactory && !ICreateTheFactory() ) |
|
return nil; |
|
|
|
return theFactory->ICreate(hClass); |
|
} |
|
|
|
|
|
|
|
uint16_t plFactory::GetNumClasses() |
|
{ |
|
if( !theFactory && !ICreateTheFactory() ) |
|
return 0; |
|
|
|
return theFactory->IGetNumClasses(); |
|
} |
|
|
|
hsBool plFactory::IDerivesFrom(uint16_t hBase, uint16_t hDer) |
|
{ |
|
if( hDer >= fCreators.GetCount() ) |
|
return false; |
|
|
|
return fCreators[hDer] ? fCreators[hDer]->HasBaseClass(hBase) : false; |
|
} |
|
|
|
hsBool plFactory::DerivesFrom(uint16_t hBase, uint16_t hDer) |
|
{ |
|
if( !theFactory && !ICreateTheFactory() ) |
|
return 0; |
|
|
|
return theFactory->IDerivesFrom(hBase, hDer); |
|
} |
|
|
|
// slow lookup for things like console |
|
uint16_t plFactory::FindClassIndex(const char* className) |
|
{ |
|
int numClasses=GetNumClasses(); |
|
|
|
if (className && theFactory) |
|
{ |
|
int i; |
|
for( i = 0; i < theFactory->fCreators.GetCount(); i++ ) |
|
{ |
|
if( theFactory->fCreators[i] && !stricmp(className, theFactory->fCreators[i]->ClassName()) ) |
|
{ |
|
return theFactory->fCreators[i]->ClassIndex(); |
|
} |
|
} |
|
} |
|
return numClasses; // err |
|
} |
|
|
|
|
|
hsBool plFactory::IIsValidClassIndex(uint16_t hClass) |
|
{ |
|
return ( hClass < fCreators.GetCount() ); |
|
} |
|
|
|
hsBool plFactory::IsValidClassIndex(uint16_t hClass) |
|
{ |
|
return theFactory->IIsValidClassIndex(hClass); |
|
} |
|
|
|
void plFactory::SetTheFactory(plFactory* fac) |
|
{ |
|
// There are four cases here. |
|
// 1) Our factory is nil, and we're being given one to use |
|
// Just take it and ref it. |
|
// 2) Our factory is non-nil, and we're being given on to use |
|
// Ours is bogus, pitch it and use the new one. |
|
// 3) Our factory is non-nil, and we're being given a nil one |
|
// Means we're being shut down. Unref the old one. If |
|
// the refcnt drops to one, we're the last one out, so |
|
// go ahead and delete it. |
|
// 4) Our factory is nil and the new one is nil |
|
// Shouldn't happen, but if it does, just ignore it. |
|
if( !theFactory && fac ) |
|
{ |
|
hsRefCnt_SafeAssign(theFactory, fac); |
|
} |
|
else if( theFactory && fac ) |
|
{ |
|
theFactory->IForceShutdown(); |
|
hsRefCnt_SafeAssign(theFactory, fac); |
|
} |
|
else if( theFactory && !fac ) |
|
{ |
|
hsRefCnt_SafeUnRef(theFactory); |
|
if( theFactory->RefCnt() < 2 ) |
|
delete theFactory; |
|
theFactory = nil; |
|
} |
|
|
|
} |
|
|
|
plFactory* plFactory::GetTheFactory() |
|
{ |
|
if( !theFactory && !ICreateTheFactory() ) |
|
return nil; |
|
|
|
return theFactory; |
|
} |
|
|
|
// For my own nefarious purposes... hsStatusMessage plCreatableIndex |
|
const char *plFactory::GetNameOfClass(uint16_t type) |
|
{ |
|
if( type < GetNumClasses() ) |
|
{ |
|
if( theFactory->fCreators[ type ] ) |
|
return theFactory->fCreators[ type ]->ClassName(); |
|
|
|
static int keyedStringsSize = sizeof(plCreatableStrings::fKeyedStrings)/4; |
|
// If we don't have a creator yet, try falling back on plCreatableStrings |
|
if( type < KEYED_OBJ_DELINEATOR && type<keyedStringsSize) |
|
return plCreatableStrings::fKeyedStrings[ type ]; |
|
|
|
if (type < plCreatableIndex::kDatabaseStructIndexesStart) |
|
{ |
|
static int nonKeyedStringsSize = sizeof(plCreatableStrings::fNonKeyedStrings)/4; |
|
int idx=type - KEYED_OBJ_DELINEATOR; |
|
if (idx<nonKeyedStringsSize) |
|
return plCreatableStrings::fNonKeyedStrings[ idx ]; |
|
} |
|
|
|
static int nonKeyedPostDBStringsSize = sizeof(plCreatableStrings::fNonKeyedPostDBStrings)/4; |
|
int idx=type - plCreatableIndex::kDatabaseStructIndexesEnd -1; |
|
if (idx<nonKeyedPostDBStringsSize) |
|
return plCreatableStrings::fNonKeyedPostDBStrings[ idx ]; |
|
} |
|
|
|
hsAssert(type < GetNumClasses() || type==0xffff,"InValid type"); |
|
return nil; |
|
} |
|
|
|
#ifdef HS_DEBUGGING |
|
|
|
/* |
|
** |
|
** Function Name: Validate |
|
** Input(s): Void |
|
** Output(s): Void |
|
** Function Description: This function examines all the Workers in the Factory and compares their Factory |
|
** index to the Enums found in plCreatableIndex. If they are Keyed objects, and |
|
** larger than 512 on the Factory index, or non-Keyed objects with a Factory |
|
** index of less than 512, exit with an Error Message. Otherwise continue through |
|
** the iteration of Factory Indices. |
|
** |
|
** |
|
*/ |
|
|
|
void plFactory::IValidate(uint16_t keyIndex) |
|
{ |
|
|
|
int FactoryIndex = GetNumClasses(); |
|
|
|
hsBool bogus = false; |
|
|
|
for(int iter=0; iter < FactoryIndex; iter++) |
|
{ |
|
if (IDerivesFrom(keyIndex, iter)) |
|
{ |
|
if(iter >= KEYED_OBJ_DELINEATOR && theFactory->fCreators[iter]) |
|
{ |
|
char Buffer[512]; |
|
sprintf(Buffer, "Object %s is a hsKeyedObject, Must appear before 'KEYED_OBJ_DELINEATOR' in plCreatableIndex.h\n",GetNameOfClass(iter)); |
|
hsStatusMessage(Buffer); |
|
bogus = true; |
|
} |
|
|
|
} |
|
else |
|
{ |
|
if(iter < KEYED_OBJ_DELINEATOR && theFactory->fCreators[iter]) |
|
{ |
|
char Buffer[512]; |
|
sprintf(Buffer, "Object %s is NOT a hsKeyedObject, Must appear after 'KEYED_OBJ_DELINEATOR' in plCreatableIndex.h\n",GetNameOfClass(iter)); |
|
hsStatusMessage(Buffer); |
|
bogus = true; |
|
|
|
} |
|
} |
|
} |
|
hsAssert(!bogus,"The class(s) you just added to plCreatableIndex.h in wrong spot, see output window"); |
|
|
|
} |
|
|
|
void plFactory::Validate(uint16_t keyIndex) |
|
{ |
|
theFactory->IValidate(keyIndex); |
|
|
|
} |
|
|
|
|
|
#endif
|
|
|