Implemented BLooper::check_lock().

Use AssertLock() more often instead of varying debugger messages.
Cleanup.


git-svn-id: file:///srv/svn/repos/haiku/haiku/trunk@13167 a95241bf-73f2-0310-859d-f6bbb57e9c96
This commit is contained in:
Axel Dörfler 2005-06-16 02:56:03 +00:00
parent 1ae95ac872
commit 410d5c37d5

View File

@ -309,13 +309,17 @@ BLooper::MessageReceived(BMessage *msg)
// saying it does nothing. Investigate.
BHandler::MessageReceived(msg);
}
//------------------------------------------------------------------------------
BMessage* BLooper::CurrentMessage() const
BMessage*
BLooper::CurrentMessage() const
{
return fLastMessage;
}
//------------------------------------------------------------------------------
BMessage* BLooper::DetachCurrentMessage()
BMessage*
BLooper::DetachCurrentMessage()
{
Lock();
BMessage* msg = fLastMessage;
@ -324,24 +328,22 @@ BMessage* BLooper::DetachCurrentMessage()
Unlock();
return msg;
}
//------------------------------------------------------------------------------
BMessageQueue* BLooper::MessageQueue() const
BMessageQueue*
BLooper::MessageQueue() const
{
return fQueue;
}
//------------------------------------------------------------------------------
bool BLooper::IsMessageWaiting() const
bool
BLooper::IsMessageWaiting() const
{
if (!IsLocked())
{
debugger("The Looper must be locked before calling IsMsgWaiting");
return false;
}
AssertLocked();
if (!fQueue->IsEmpty())
{
return true;
}
/**
@note: What we're doing here differs slightly from the R5 implementation.
@ -355,65 +357,49 @@ bool BLooper::IsMessageWaiting() const
where I come from. ;)
*/
int32 count;
do
{
do {
count = port_buffer_size_etc(fMsgPort, B_RELATIVE_TIMEOUT, 0);
} while (count == B_INTERRUPTED);
return count > 0;
}
//------------------------------------------------------------------------------
void BLooper::AddHandler(BHandler* handler)
void
BLooper::AddHandler(BHandler* handler)
{
if (!handler)
{
if (handler == NULL)
return;
}
if (!IsLocked())
{
debugger("Looper must be locked before calling AddHandler.");
}
AssertLocked();
if (handler->Looper() == NULL)
{
if (handler->Looper() == NULL) {
fHandlers.AddItem(handler);
handler->SetLooper(this);
if (handler != this) // avoid a cycle
handler->SetNextHandler(this);
BList* Filters = handler->FilterList();
if (Filters)
{
for (int32 i = 0; i < Filters->CountItems(); ++i)
{
((BMessageFilter*)Filters->ItemAt(i))->SetLooper(this);
BList* filters = handler->FilterList();
if (filters) {
for (int32 i = 0; i < filters->CountItems(); ++i) {
((BMessageFilter*)filters->ItemAt(i))->SetLooper(this);
}
}
}
}
//------------------------------------------------------------------------------
bool BLooper::RemoveHandler(BHandler* handler)
bool
BLooper::RemoveHandler(BHandler* handler)
{
// R5 implementation didn't bother to check its params, thus NULL handlers
// will seg fault. Bad form, you know.
if (!handler)
{
if (handler == NULL)
return false;
}
// Correction; testing shows the looper *does* need to be locked for this;
// it just doesn't use AssertLocked() for that.
if (!IsLocked())
{
debugger("Looper must be locked before calling RemoveHandler.");
}
AssertLocked();
if (handler->Looper() == this && fHandlers.RemoveItem(handler))
{
if (handler->Looper() == this && fHandlers.RemoveItem(handler)) {
if (handler == fPreferred)
{
fPreferred = NULL;
}
handler->SetNextHandler(NULL);
handler->SetLooper(NULL);
@ -422,47 +408,48 @@ bool BLooper::RemoveHandler(BHandler* handler)
return false;
}
//------------------------------------------------------------------------------
int32 BLooper::CountHandlers() const
int32
BLooper::CountHandlers() const
{
AssertLocked();
return fHandlers.CountItems();
}
//------------------------------------------------------------------------------
BHandler* BLooper::HandlerAt(int32 index) const
BHandler*
BLooper::HandlerAt(int32 index) const
{
if (!IsLocked())
{
debugger("Looper must be locked before calling HandlerAt.");
}
AssertLocked();
return (BHandler*)fHandlers.ItemAt(index);
}
//------------------------------------------------------------------------------
int32 BLooper::IndexOf(BHandler* handler) const
int32
BLooper::IndexOf(BHandler* handler) const
{
if (!IsLocked())
{
debugger("Looper must be locked before calling IndexOf.");
}
AssertLocked();
return fHandlers.IndexOf(handler);
}
//------------------------------------------------------------------------------
BHandler* BLooper::PreferredHandler() const
BHandler*
BLooper::PreferredHandler() const
{
return fPreferred;
}
//------------------------------------------------------------------------------
void BLooper::SetPreferredHandler(BHandler* handler)
void
BLooper::SetPreferredHandler(BHandler* handler)
{
if (handler && handler->Looper() == this && IndexOf(handler) >= 0)
{
if (handler && handler->Looper() == this && IndexOf(handler) >= 0) {
fPreferred = handler;
}
else
{
} else {
fPreferred = NULL;
}
}
@ -514,8 +501,7 @@ BLooper::Quit()
PRINT((" is locked\n"));
if (!fRunCalled)
{
if (!fRunCalled) {
PRINT((" Run() has not been called yet\n"));
fTerminating = true;
delete this;
@ -574,8 +560,10 @@ BLooper::Lock()
// Defer to global _Lock(); see notes there
return _Lock(this, -1, B_INFINITE_TIMEOUT) == B_OK;
}
//------------------------------------------------------------------------------
void BLooper::Unlock()
void
BLooper::Unlock()
{
PRINT(("BLooper::Unlock()\n"));
// Make sure we're locked to begin with
@ -585,8 +573,7 @@ PRINT(("BLooper::Unlock()\n"));
--fOwnerCount;
PRINT((" fOwnerCount now: %ld\n", fOwnerCount));
// Check to see if the owner still wants a lock
if (fOwnerCount == 0)
{
if (fOwnerCount == 0) {
// Set fOwner to invalid thread_id (< 0)
fOwner = -1;
@ -604,21 +591,21 @@ PRINT((" fAtomicCount now: %ld\n", fAtomicCount));
}
PRINT(("BLooper::Unlock() done\n"));
}
//------------------------------------------------------------------------------
bool BLooper::IsLocked() const
bool
BLooper::IsLocked() const
{
// We have to lock the list for the call to IsLooperValid(). Has the side
// effect of not letting the looper get deleted while we're here.
BObjectLocker<BLooperList> ListLock(gLooperList);
if (!ListLock.IsLocked())
{
if (!ListLock.IsLocked()) {
// If we can't lock the list, our semaphore is probably toast
return false;
}
if (!IsLooperValid(this))
{
if (!IsLooperValid(this)) {
// The looper is gone, so of course it's not locked
return false;
}
@ -626,56 +613,71 @@ bool BLooper::IsLocked() const
// Got this from Jeremy's BLocker implementation
return find_thread(NULL) == fOwner;
}
//------------------------------------------------------------------------------
status_t BLooper::LockWithTimeout(bigtime_t timeout)
status_t
BLooper::LockWithTimeout(bigtime_t timeout)
{
return _Lock(this, -1, timeout);
}
//------------------------------------------------------------------------------
thread_id BLooper::Thread() const
thread_id
BLooper::Thread() const
{
return fTaskID;
}
//------------------------------------------------------------------------------
team_id BLooper::Team() const
team_id
BLooper::Team() const
{
return sTeamID;
}
//------------------------------------------------------------------------------
BLooper* BLooper::LooperForThread(thread_id tid)
BLooper*
BLooper::LooperForThread(thread_id tid)
{
BObjectLocker<BLooperList> ListLock(gLooperList);
if (ListLock.IsLocked())
{
return gLooperList.LooperForThread(tid);
}
return NULL;
}
//------------------------------------------------------------------------------
thread_id BLooper::LockingThread() const
thread_id
BLooper::LockingThread() const
{
return fOwner;
}
//------------------------------------------------------------------------------
int32 BLooper::CountLocks() const
int32
BLooper::CountLocks() const
{
return fOwnerCount;
}
//------------------------------------------------------------------------------
int32 BLooper::CountLockRequests() const
int32
BLooper::CountLockRequests() const
{
return fAtomicCount;
}
//------------------------------------------------------------------------------
sem_id BLooper::Sem() const
sem_id
BLooper::Sem() const
{
return fLockSem;
}
//------------------------------------------------------------------------------
BHandler* BLooper::ResolveSpecifier(BMessage* msg, int32 index,
BMessage* specifier, int32 form,
const char* property)
BHandler*
BLooper::ResolveSpecifier(BMessage* msg, int32 index,
BMessage* specifier, int32 form, const char* property)
{
/**
@note When I was first dumping the results of GetSupportedSuites() from
@ -693,29 +695,23 @@ BHandler* BLooper::ResolveSpecifier(BMessage* msg, int32 index,
uint32 data;
status_t err = B_OK;
const char* errMsg = "";
if (PropertyInfo.FindMatch(msg, index, specifier, form, property, &data) >= 0)
{
switch (data)
{
if (PropertyInfo.FindMatch(msg, index, specifier, form, property, &data) >= 0) {
switch (data) {
case BLOOPER_PROCESS_INTERNALLY:
return this;
case BLOOPER_HANDLER_BY_INDEX:
{
int32 index = specifier->FindInt32("index");
if (form == B_REVERSE_INDEX_SPECIFIER)
{
if (form == B_REVERSE_INDEX_SPECIFIER) {
index = CountHandlers() - index;
}
BHandler* target = HandlerAt(index);
if (target)
{
if (target) {
// Specifier has been fully handled
msg->PopSpecifier();
return target;
}
else
{
} else {
err = B_BAD_INDEX;
errMsg = "handler index out of range";
}
@ -727,11 +723,9 @@ BHandler* BLooper::ResolveSpecifier(BMessage* msg, int32 index,
errMsg = "Didn't understand the specifier(s)";
break;
}
}
else
{
} else {
return BHandler::ResolveSpecifier(msg, index, specifier, form,
property);
property);
}
BMessage Reply(B_MESSAGE_NOT_UNDERSTOOD);
@ -741,84 +735,65 @@ BHandler* BLooper::ResolveSpecifier(BMessage* msg, int32 index,
return NULL;
}
//------------------------------------------------------------------------------
status_t BLooper::GetSupportedSuites(BMessage* data)
status_t
BLooper::GetSupportedSuites(BMessage* data)
{
status_t err = B_OK;
if (data == NULL)
return B_BAD_VALUE;
if (!data)
{
err = B_BAD_VALUE;
status_t status = data->AddString("Suites", "suite/vnd.Be-handler");
if (status == B_OK) {
BPropertyInfo PropertyInfo(gLooperPropInfo);
status = data->AddFlat("message", &PropertyInfo);
if (status == B_OK)
status = BHandler::GetSupportedSuites(data);
}
if (!err)
{
err = data->AddString("Suites", "suite/vnd.Be-handler");
if (!err)
{
BPropertyInfo PropertyInfo(gLooperPropInfo);
err = data->AddFlat("message", &PropertyInfo);
if (!err)
{
err = BHandler::GetSupportedSuites(data);
}
}
}
return err;
return status;
}
//------------------------------------------------------------------------------
void BLooper::AddCommonFilter(BMessageFilter* filter)
void
BLooper::AddCommonFilter(BMessageFilter* filter)
{
if (!filter)
{
return;
}
if (!IsLocked())
{
debugger("Owning Looper must be locked before calling AddCommonFilter");
return;
}
AssertLocked();
if (filter->Looper())
{
if (filter->Looper()) {
debugger("A MessageFilter can only be used once.");
return;
}
if (!fCommonFilters)
{
fCommonFilters = new BList(FILTER_LIST_BLOCK_SIZE);
}
filter->SetLooper(this);
fCommonFilters->AddItem(filter);
}
//------------------------------------------------------------------------------
bool BLooper::RemoveCommonFilter(BMessageFilter* filter)
bool
BLooper::RemoveCommonFilter(BMessageFilter* filter)
{
if (!IsLocked())
{
debugger("Owning Looper must be locked before calling "
"RemoveCommonFilter");
return false;
}
AssertLocked();
if (!fCommonFilters)
{
return false;
}
bool result = fCommonFilters->RemoveItem(filter);
if (result)
{
filter->SetLooper(NULL);
}
return result;
}
//------------------------------------------------------------------------------
void BLooper::SetCommonFilterList(BList* filters)
void
BLooper::SetCommonFilterList(BList* filters)
{
// We have a somewhat serious problem here. It is entirely possible in R5
// to assign a given list of filters to *two* BLoopers simultaneously. This
@ -827,32 +802,22 @@ void BLooper::SetCommonFilterList(BList* filters)
// has already been deleted. In R5, this results in a general protection
// fault. We fix this by checking the filter list for ownership issues.
if (!IsLocked())
{
debugger("Owning Looper must be locked before calling "
"SetCommonFilterList");
return;
}
AssertLocked();
BMessageFilter* filter;
if (filters)
{
if (filters) {
// Check for ownership issues
for (int32 i = 0; i < filters->CountItems(); ++i)
{
for (int32 i = 0; i < filters->CountItems(); ++i) {
filter = (BMessageFilter*)filters->ItemAt(i);
if (filter->Looper())
{
if (filter->Looper()) {
debugger("A MessageFilter can only be used once.");
return;
}
}
}
if (fCommonFilters)
{
for (int32 i = 0; i < fCommonFilters->CountItems(); ++i)
{
if (fCommonFilters) {
for (int32 i = 0; i < fCommonFilters->CountItems(); ++i) {
delete (BMessageFilter*)fCommonFilters->ItemAt(i);
}
@ -862,67 +827,58 @@ void BLooper::SetCommonFilterList(BList* filters)
// Per the BeBook, we take ownership of the list
fCommonFilters = filters;
if (fCommonFilters)
{
for (int32 i = 0; i < fCommonFilters->CountItems(); ++i)
{
if (fCommonFilters) {
for (int32 i = 0; i < fCommonFilters->CountItems(); ++i) {
filter = (BMessageFilter*)fCommonFilters->ItemAt(i);
filter->SetLooper(this);
}
}
}
//------------------------------------------------------------------------------
BList* BLooper::CommonFilterList() const
BList*
BLooper::CommonFilterList() const
{
return fCommonFilters;
}
//------------------------------------------------------------------------------
status_t BLooper::Perform(perform_code d, void* arg)
status_t
BLooper::Perform(perform_code d, void* arg)
{
// This is sort of what we're doing for this function everywhere
return BHandler::Perform(d, arg);
}
//------------------------------------------------------------------------------
BMessage* BLooper::MessageFromPort(bigtime_t timeout)
BMessage*
BLooper::MessageFromPort(bigtime_t timeout)
{
return ReadMessageFromPort(timeout);
}
//------------------------------------------------------------------------------
void BLooper::_ReservedLooper1()
{
}
//------------------------------------------------------------------------------
void BLooper::_ReservedLooper2()
{
}
//------------------------------------------------------------------------------
void BLooper::_ReservedLooper3()
{
}
//------------------------------------------------------------------------------
void BLooper::_ReservedLooper4()
{
}
//------------------------------------------------------------------------------
void BLooper::_ReservedLooper5()
{
}
//------------------------------------------------------------------------------
void BLooper::_ReservedLooper6()
{
}
//------------------------------------------------------------------------------
void BLooper::_ReservedLooper1() {}
void BLooper::_ReservedLooper2() {}
void BLooper::_ReservedLooper3() {}
void BLooper::_ReservedLooper4() {}
void BLooper::_ReservedLooper5() {}
void BLooper::_ReservedLooper6() {}
BLooper::BLooper(const BLooper&)
{
// Copy construction not allowed
}
//------------------------------------------------------------------------------
BLooper& BLooper::operator=(const BLooper& )
{
// Looper copying not allowed
return *this;
}
//------------------------------------------------------------------------------
BLooper::BLooper(int32 priority, port_id port, const char* name)
{
// This must be a legacy constructor
@ -1508,7 +1464,11 @@ BLooper::apply_filters(BList* list, BMessage* msg, BHandler* target)
void
BLooper::check_lock()
{
// NOTE: any use for this?
// This is a cheap variant of AssertLocked()
// It is used in situations where it's clear that the looper is valid,
// ie. from handlers
if (fOwner == -1 || fOwner != find_thread(NULL))
debugger("Looper must be locked.");
}