#include <DirectMessageTarget.h>
#include <TokenSpace.h>
#include <PthreadMutexLocker.h>
namespace BPrivate {
BTokenSpace gDefaultTokens;
static int32
get_next_token(int32 token)
{
if (token == INT32_MAX)
return 1;
else
return token + 1;
}
BTokenSpace::BTokenSpace()
:
fNextToken(1)
{
fLock = (pthread_mutex_t)PTHREAD_RECURSIVE_MUTEX_INITIALIZER_NP;
}
BTokenSpace::~BTokenSpace()
{
pthread_mutex_destroy(&fLock);
}
int32
BTokenSpace::NewToken(int16 type, void* object)
{
PthreadMutexLocker locker(&fLock);
token_info tokenInfo = { type, object, NULL };
int32 wraparoundToken = fNextToken;
try {
for (;;) {
int32 token = fNextToken;
bool done = fTokenMap.insert(std::make_pair(token, tokenInfo)).second;
fNextToken = get_next_token(token);
if (done)
return token;
if (fNextToken == wraparoundToken)
return -1;
}
} catch (std::bad_alloc& exception) {
return -1;
}
}
bool
BTokenSpace::SetToken(int32 token, int16 type, void* object)
{
PthreadMutexLocker locker(&fLock);
token_info tokenInfo = { type, object, NULL };
try {
fTokenMap[token] = tokenInfo;
} catch (std::bad_alloc& exception) {
return false;
}
if (token >= fNextToken)
fNextToken = get_next_token(token);
return true;
}
bool
BTokenSpace::RemoveToken(int32 token)
{
PthreadMutexLocker locker(&fLock);
TokenMap::iterator iterator = fTokenMap.find(token);
if (iterator == fTokenMap.end())
return false;
fTokenMap.erase(iterator);
return true;
}
bool
BTokenSpace::CheckToken(int32 token, int16 type) const
{
PthreadMutexLocker locker(&fLock);
TokenMap::const_iterator iterator = fTokenMap.find(token);
if (iterator != fTokenMap.end() && iterator->second.type == type)
return true;
return false;
}
status_t
BTokenSpace::GetToken(int32 token, int16 type, void** _object) const
{
if (token < 1)
return B_ENTRY_NOT_FOUND;
PthreadMutexLocker locker(&fLock);
TokenMap::const_iterator iterator = fTokenMap.find(token);
if (iterator == fTokenMap.end() || iterator->second.type != type)
return B_ENTRY_NOT_FOUND;
*_object = iterator->second.object;
return B_OK;
}
status_t
BTokenSpace::SetHandlerTarget(int32 token, BDirectMessageTarget* target)
{
if (token < 1)
return B_ENTRY_NOT_FOUND;
PthreadMutexLocker locker(&fLock);
TokenMap::iterator iterator = fTokenMap.find(token);
if (iterator == fTokenMap.end() || iterator->second.type != B_HANDLER_TOKEN)
return B_ENTRY_NOT_FOUND;
if (iterator->second.target != NULL)
iterator->second.target->Release();
iterator->second.target = target;
if (target != NULL)
target->Acquire();
return B_OK;
}
status_t
BTokenSpace::AcquireHandlerTarget(int32 token, BDirectMessageTarget** _target)
{
if (token < 1)
return B_ENTRY_NOT_FOUND;
PthreadMutexLocker locker(&fLock);
TokenMap::const_iterator iterator = fTokenMap.find(token);
if (iterator == fTokenMap.end() || iterator->second.type != B_HANDLER_TOKEN)
return B_ENTRY_NOT_FOUND;
if (iterator->second.target != NULL)
iterator->second.target->Acquire();
*_target = iterator->second.target;
return B_OK;
}
}