* Change get_next_sem_info() to be index based, not ID based - the semaphore IDs
aren't monotonically increasing which this code was assuming. This fixes bug #4917. git-svn-id: file:///srv/svn/repos/haiku/haiku/trunk@34074 a95241bf-73f2-0310-859d-f6bbb57e9c96
This commit is contained in:
parent
9b716314fc
commit
07d4556346
@ -1123,47 +1123,39 @@ _get_next_sem_info(team_id teamID, int32 *_cookie, struct sem_info *info,
|
|||||||
if (team == NULL)
|
if (team == NULL)
|
||||||
return B_BAD_TEAM_ID;
|
return B_BAD_TEAM_ID;
|
||||||
|
|
||||||
int32 id = *_cookie;
|
// TODO: find a way to iterate the list that is more reliable
|
||||||
sem_entry* sem = NULL;
|
sem_entry* sem = (sem_entry*)list_get_first_item(&team->sem_list);
|
||||||
|
int32 newIndex = *_cookie;
|
||||||
if (id != 0) {
|
int32 index = 0;
|
||||||
// shortcut to the first entry
|
|
||||||
sem = &sSems[id % sMaxSems];
|
|
||||||
GRAB_SEM_LOCK(*sem);
|
|
||||||
|
|
||||||
// Check if the semaphore got deleted or reused in the mean time
|
|
||||||
if (sem->id != id)
|
|
||||||
sem = NULL;
|
|
||||||
|
|
||||||
RELEASE_SEM_LOCK(*sem);
|
|
||||||
}
|
|
||||||
if (sem == NULL)
|
|
||||||
sem = (sem_entry*)list_get_first_item(&team->sem_list);
|
|
||||||
|
|
||||||
bool found = false;
|
bool found = false;
|
||||||
|
|
||||||
while (!found) {
|
while (!found) {
|
||||||
// find the next entry to be returned
|
// find the next entry to be returned
|
||||||
while (sem != NULL && id >= sem->id)
|
while (sem != NULL && index < newIndex) {
|
||||||
sem = (sem_entry*)list_get_next_item(&team->sem_list, sem);
|
sem = (sem_entry*)list_get_next_item(&team->sem_list, sem);
|
||||||
|
index++;
|
||||||
|
}
|
||||||
|
|
||||||
if (sem == NULL)
|
if (sem == NULL)
|
||||||
return B_BAD_VALUE;
|
return B_BAD_VALUE;
|
||||||
|
|
||||||
GRAB_SEM_LOCK(*sem);
|
GRAB_SEM_LOCK(*sem);
|
||||||
|
|
||||||
if (sem->id != -1 && sem->u.used.owner == team) {
|
if (sem->id != -1 && sem->u.used.owner == team) {
|
||||||
// found one!
|
// found one!
|
||||||
fill_sem_info(sem, info, size);
|
fill_sem_info(sem, info, size);
|
||||||
id = sem->id;
|
newIndex = index + 1;
|
||||||
found = true;
|
found = true;
|
||||||
}
|
} else
|
||||||
|
newIndex++;
|
||||||
|
|
||||||
RELEASE_SEM_LOCK(*sem);
|
RELEASE_SEM_LOCK(*sem);
|
||||||
}
|
}
|
||||||
|
|
||||||
if (!found)
|
if (!found)
|
||||||
return B_BAD_VALUE;
|
return B_BAD_VALUE;
|
||||||
|
|
||||||
*_cookie = id;
|
*_cookie = newIndex;
|
||||||
return B_OK;
|
return B_OK;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
Loading…
Reference in New Issue
Block a user