2004-02-23 06:31:08 +03:00
|
|
|
/* Semaphore code. Lots of "todo" items */
|
|
|
|
|
|
|
|
/*
|
|
|
|
** Copyright 2002-2004, The OpenBeOS Team. All rights reserved.
|
|
|
|
** Distributed under the terms of the OpenBeOS License.
|
|
|
|
*/
|
2002-07-09 16:24:59 +04:00
|
|
|
|
|
|
|
/*
|
|
|
|
** Copyright 2001, Travis Geiselbrecht. All rights reserved.
|
|
|
|
** Distributed under the terms of the NewOS License.
|
|
|
|
*/
|
2002-10-30 02:07:06 +03:00
|
|
|
|
2002-07-09 16:24:59 +04:00
|
|
|
#include <kernel.h>
|
|
|
|
#include <OS.h>
|
|
|
|
#include <sem.h>
|
|
|
|
#include <smp.h>
|
|
|
|
#include <int.h>
|
|
|
|
#include <arch/int.h>
|
|
|
|
#include <timer.h>
|
|
|
|
#include <debug.h>
|
2002-10-30 02:07:06 +03:00
|
|
|
#include <malloc.h>
|
2002-07-09 16:24:59 +04:00
|
|
|
#include <thread.h>
|
|
|
|
#include <Errors.h>
|
|
|
|
|
2003-10-08 03:12:37 +04:00
|
|
|
#include <boot/kernel_args.h>
|
2002-07-09 16:24:59 +04:00
|
|
|
|
|
|
|
#include <string.h>
|
|
|
|
#include <stdlib.h>
|
|
|
|
|
2002-09-30 07:43:32 +04:00
|
|
|
#define TRACE_SEM 0
|
|
|
|
#if TRACE_SEM
|
|
|
|
# define TRACE(x) dprintf x
|
|
|
|
# define TRACE_BLOCK(x) dprintf x
|
|
|
|
#else
|
|
|
|
# define TRACE(x) ;
|
|
|
|
# define TRACE_BLOCK(x) ;
|
|
|
|
#endif
|
|
|
|
|
2002-07-09 16:24:59 +04:00
|
|
|
struct sem_entry {
|
2002-10-26 20:13:36 +04:00
|
|
|
sem_id id;
|
2003-08-04 04:06:54 +04:00
|
|
|
spinlock lock; // protects only the id field when unused
|
|
|
|
union {
|
|
|
|
// when slot in use
|
|
|
|
struct {
|
|
|
|
int count;
|
|
|
|
struct thread_queue q;
|
|
|
|
char *name;
|
|
|
|
team_id owner; // if set to -1, means owned by a port
|
|
|
|
} used;
|
|
|
|
|
|
|
|
// when slot unused
|
|
|
|
struct {
|
|
|
|
sem_id next_id;
|
|
|
|
struct sem_entry *next;
|
|
|
|
} unused;
|
|
|
|
} u;
|
2002-07-09 16:24:59 +04:00
|
|
|
};
|
|
|
|
|
2003-08-04 04:06:54 +04:00
|
|
|
// Todo: Compute based on the amount of available memory.
|
2002-07-09 16:24:59 +04:00
|
|
|
#define MAX_SEMS 4096
|
|
|
|
|
2002-09-30 07:43:32 +04:00
|
|
|
static struct sem_entry *gSems = NULL;
|
|
|
|
static region_id gSemRegion = 0;
|
|
|
|
static bool gSemsActive = false;
|
2003-08-04 04:06:54 +04:00
|
|
|
static struct sem_entry *gFreeSemsHead = NULL;
|
|
|
|
static struct sem_entry *gFreeSemsTail = NULL;
|
2002-07-09 16:24:59 +04:00
|
|
|
|
2002-10-26 20:13:36 +04:00
|
|
|
static spinlock sem_spinlock = 0;
|
2002-07-21 02:52:04 +04:00
|
|
|
#define GRAB_SEM_LIST_LOCK() acquire_spinlock(&sem_spinlock)
|
|
|
|
#define RELEASE_SEM_LIST_LOCK() release_spinlock(&sem_spinlock)
|
|
|
|
#define GRAB_SEM_LOCK(s) acquire_spinlock(&(s).lock)
|
|
|
|
#define RELEASE_SEM_LOCK(s) release_spinlock(&(s).lock)
|
2002-07-09 16:24:59 +04:00
|
|
|
|
|
|
|
static int remove_thread_from_sem(struct thread *t, struct sem_entry *sem, struct thread_queue *queue, int sem_errcode);
|
|
|
|
|
|
|
|
struct sem_timeout_args {
|
|
|
|
thread_id blocked_thread;
|
|
|
|
sem_id blocked_sem_id;
|
|
|
|
int sem_count;
|
|
|
|
};
|
|
|
|
|
2002-08-05 09:23:23 +04:00
|
|
|
|
|
|
|
static int
|
|
|
|
dump_sem_list(int argc, char **argv)
|
2002-07-09 16:24:59 +04:00
|
|
|
{
|
|
|
|
int i;
|
|
|
|
|
2002-09-30 07:43:32 +04:00
|
|
|
for (i = 0; i < MAX_SEMS; i++) {
|
|
|
|
if (gSems[i].id >= 0)
|
2003-08-04 04:06:54 +04:00
|
|
|
dprintf("%p\tid: 0x%lx\t\tname: '%s'\n", &gSems[i], gSems[i].id,
|
|
|
|
gSems[i].u.used.name);
|
2002-07-09 16:24:59 +04:00
|
|
|
}
|
2002-07-18 02:07:37 +04:00
|
|
|
return 0;
|
2002-07-09 16:24:59 +04:00
|
|
|
}
|
|
|
|
|
2002-08-05 09:23:23 +04:00
|
|
|
|
|
|
|
static void
|
2002-09-30 07:43:32 +04:00
|
|
|
dump_sem(struct sem_entry *sem)
|
2002-07-09 16:24:59 +04:00
|
|
|
{
|
2003-08-04 04:06:54 +04:00
|
|
|
dprintf("SEM: %p\n", sem);
|
|
|
|
dprintf("id: %ld\n", sem->id);
|
|
|
|
if (sem->id >= 0) {
|
|
|
|
dprintf("name: '%s'\n", sem->u.used.name);
|
|
|
|
dprintf("owner: 0x%lx\n", sem->u.used.owner);
|
|
|
|
dprintf("count: 0x%x\n", sem->u.used.count);
|
|
|
|
dprintf("queue: head %p tail %p\n", sem->u.used.q.head,
|
|
|
|
sem->u.used.q.tail);
|
|
|
|
} else {
|
|
|
|
dprintf("next: %p\n", sem->u.unused.next);
|
|
|
|
dprintf("next_id: %ld\n", sem->u.unused.next_id);
|
|
|
|
}
|
2002-07-09 16:24:59 +04:00
|
|
|
}
|
|
|
|
|
2002-08-05 09:23:23 +04:00
|
|
|
|
|
|
|
static int
|
|
|
|
dump_sem_info(int argc, char **argv)
|
2002-07-09 16:24:59 +04:00
|
|
|
{
|
|
|
|
int i;
|
|
|
|
|
2002-07-21 02:52:04 +04:00
|
|
|
if (argc < 2) {
|
2002-07-09 16:24:59 +04:00
|
|
|
dprintf("sem: not enough arguments\n");
|
2002-07-18 02:07:37 +04:00
|
|
|
return 0;
|
2002-07-09 16:24:59 +04:00
|
|
|
}
|
|
|
|
|
|
|
|
// if the argument looks like a hex number, treat it as such
|
2002-07-21 02:52:04 +04:00
|
|
|
if (strlen(argv[1]) > 2 && argv[1][0] == '0' && argv[1][1] == 'x') {
|
2002-07-09 16:24:59 +04:00
|
|
|
unsigned long num = atoul(argv[1]);
|
|
|
|
|
2002-07-21 02:52:04 +04:00
|
|
|
if (num > KERNEL_BASE && num <= (KERNEL_BASE + (KERNEL_SIZE - 1))) {
|
2002-07-09 16:24:59 +04:00
|
|
|
// XXX semi-hack
|
2002-09-30 07:43:32 +04:00
|
|
|
dump_sem((struct sem_entry *)num);
|
2002-07-18 02:07:37 +04:00
|
|
|
return 0;
|
2002-09-30 07:43:32 +04:00
|
|
|
} else {
|
2002-07-09 16:24:59 +04:00
|
|
|
unsigned slot = num % MAX_SEMS;
|
2002-09-30 07:43:32 +04:00
|
|
|
if (gSems[slot].id != (int)num) {
|
2002-07-09 16:24:59 +04:00
|
|
|
dprintf("sem 0x%lx doesn't exist!\n", num);
|
2002-07-18 02:07:37 +04:00
|
|
|
return 0;
|
2002-07-09 16:24:59 +04:00
|
|
|
}
|
2002-09-30 07:43:32 +04:00
|
|
|
dump_sem(&gSems[slot]);
|
2002-07-18 02:07:37 +04:00
|
|
|
return 0;
|
2002-07-09 16:24:59 +04:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// walk through the sem list, trying to match name
|
2002-09-30 07:43:32 +04:00
|
|
|
for (i = 0; i < MAX_SEMS; i++) {
|
2003-08-04 04:06:54 +04:00
|
|
|
if (gSems[i].u.used.name != NULL
|
|
|
|
&& strcmp(argv[1], gSems[i].u.used.name) == 0) {
|
2002-09-30 07:43:32 +04:00
|
|
|
dump_sem(&gSems[i]);
|
|
|
|
return 0;
|
|
|
|
}
|
2002-07-09 16:24:59 +04:00
|
|
|
}
|
2002-08-05 09:23:23 +04:00
|
|
|
return 0;
|
2002-07-09 16:24:59 +04:00
|
|
|
}
|
|
|
|
|
2003-08-04 04:06:54 +04:00
|
|
|
/*! \brief Appends a semaphore slot to the free list.
|
|
|
|
|
|
|
|
The semaphore list must be locked.
|
|
|
|
The slot's id field is not changed. It should already be set to -1.
|
|
|
|
|
|
|
|
\param slot The index of the semaphore slot.
|
|
|
|
\param nextID The ID the slot will get when reused. If < 0 the \a slot
|
|
|
|
is used.
|
|
|
|
*/
|
|
|
|
static
|
|
|
|
void
|
|
|
|
free_sem_slot(int slot, sem_id nextID)
|
|
|
|
{
|
|
|
|
struct sem_entry *sem = gSems + slot;
|
|
|
|
// set next_id to the next possible value; for sanity check the current ID
|
|
|
|
if (nextID < 0)
|
|
|
|
sem->u.unused.next_id = slot;
|
|
|
|
else
|
|
|
|
sem->u.unused.next_id = nextID;
|
|
|
|
// append the entry to the list
|
|
|
|
if (gFreeSemsTail)
|
|
|
|
gFreeSemsTail->u.unused.next = sem;
|
|
|
|
else
|
|
|
|
gFreeSemsHead = sem;
|
|
|
|
gFreeSemsTail = sem;
|
|
|
|
sem->u.unused.next = NULL;
|
|
|
|
}
|
|
|
|
|
2002-08-05 09:23:23 +04:00
|
|
|
|
2002-09-03 06:09:48 +04:00
|
|
|
status_t
|
2002-08-05 09:23:23 +04:00
|
|
|
sem_init(kernel_args *ka)
|
2002-07-09 16:24:59 +04:00
|
|
|
{
|
|
|
|
int i;
|
|
|
|
|
2002-09-30 07:43:32 +04:00
|
|
|
TRACE(("sem_init: entry\n"));
|
2002-07-09 16:24:59 +04:00
|
|
|
|
|
|
|
// create and initialize semaphore table
|
2003-08-19 18:28:11 +04:00
|
|
|
gSemRegion = create_area("sem_table", (void **)&gSems, B_ANY_KERNEL_ADDRESS,
|
|
|
|
sizeof(struct sem_entry) * MAX_SEMS, B_FULL_LOCK, B_KERNEL_READ_AREA | B_KERNEL_WRITE_AREA);
|
2002-09-30 07:43:32 +04:00
|
|
|
if (gSemRegion < 0)
|
2002-07-09 16:24:59 +04:00
|
|
|
panic("unable to allocate semaphore table!\n");
|
|
|
|
|
2002-09-30 07:43:32 +04:00
|
|
|
memset(gSems, 0, sizeof(struct sem_entry) * MAX_SEMS);
|
2003-08-04 04:06:54 +04:00
|
|
|
for (i = 0; i < MAX_SEMS; i++) {
|
2002-09-30 07:43:32 +04:00
|
|
|
gSems[i].id = -1;
|
2003-08-04 04:06:54 +04:00
|
|
|
free_sem_slot(i, i);
|
|
|
|
}
|
2002-07-09 16:24:59 +04:00
|
|
|
|
|
|
|
// add debugger commands
|
2002-07-18 02:07:37 +04:00
|
|
|
add_debugger_command("sems", &dump_sem_list, "Dump a list of all active semaphores");
|
|
|
|
add_debugger_command("sem", &dump_sem_info, "Dump info about a particular semaphore");
|
2002-07-09 16:24:59 +04:00
|
|
|
|
2002-09-30 07:43:32 +04:00
|
|
|
TRACE(("sem_init: exit\n"));
|
2002-07-09 16:24:59 +04:00
|
|
|
|
2002-09-30 07:43:32 +04:00
|
|
|
gSemsActive = true;
|
2002-07-09 16:24:59 +04:00
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2002-09-30 07:43:32 +04:00
|
|
|
|
|
|
|
/** Creates a semaphore with the given parameters.
|
|
|
|
* Note, the team_id is not checked, it must be correct, or else
|
|
|
|
* that semaphore might not be deleted.
|
|
|
|
* This function is only available from within the kernel, and
|
|
|
|
* should not be made public - if possible, we should remove it
|
|
|
|
* completely (and have only create_sem() exported).
|
|
|
|
*/
|
|
|
|
|
|
|
|
sem_id
|
|
|
|
create_sem_etc(int32 count, const char *name, team_id owner)
|
2002-07-09 16:24:59 +04:00
|
|
|
{
|
2003-08-04 04:06:54 +04:00
|
|
|
struct sem_entry *sem = NULL;
|
2002-07-09 16:24:59 +04:00
|
|
|
int state;
|
|
|
|
sem_id retval = B_NO_MORE_SEMS;
|
|
|
|
char *temp_name;
|
2002-08-29 04:03:45 +04:00
|
|
|
int name_len;
|
2002-07-09 16:24:59 +04:00
|
|
|
|
2002-09-30 07:43:32 +04:00
|
|
|
if (gSemsActive == false)
|
2002-07-09 16:24:59 +04:00
|
|
|
return B_NO_MORE_SEMS;
|
2002-09-30 07:43:32 +04:00
|
|
|
|
2002-08-29 04:03:45 +04:00
|
|
|
if (name == NULL)
|
2002-09-11 16:23:56 +04:00
|
|
|
name = "unnamed semaphore";
|
2002-08-29 04:03:45 +04:00
|
|
|
|
2002-09-11 16:23:56 +04:00
|
|
|
name_len = strlen(name) + 1;
|
|
|
|
name_len = min(name_len, SYS_MAX_OS_NAME_LEN);
|
2002-10-30 02:07:06 +03:00
|
|
|
temp_name = (char *)malloc(name_len);
|
2002-08-29 04:03:45 +04:00
|
|
|
if (temp_name == NULL)
|
2002-09-30 07:43:32 +04:00
|
|
|
return B_NO_MEMORY;
|
2002-08-29 04:03:45 +04:00
|
|
|
strlcpy(temp_name, name, name_len);
|
2002-07-09 16:24:59 +04:00
|
|
|
|
2002-07-25 05:05:51 +04:00
|
|
|
state = disable_interrupts();
|
2002-07-09 16:24:59 +04:00
|
|
|
GRAB_SEM_LIST_LOCK();
|
|
|
|
|
2003-08-04 04:06:54 +04:00
|
|
|
// get the first slot from the free list
|
|
|
|
sem = gFreeSemsHead;
|
|
|
|
if (sem) {
|
|
|
|
// remove it from the free list
|
|
|
|
gFreeSemsHead = sem->u.unused.next;
|
|
|
|
if (!gFreeSemsHead)
|
|
|
|
gFreeSemsTail = NULL;
|
|
|
|
// init the slot
|
|
|
|
GRAB_SEM_LOCK(*sem);
|
|
|
|
sem->id = sem->u.unused.next_id;
|
|
|
|
sem->u.used.count = count;
|
|
|
|
sem->u.used.q.tail = NULL;
|
|
|
|
sem->u.used.q.head = NULL;
|
|
|
|
sem->u.used.name = temp_name;
|
|
|
|
sem->u.used.owner = owner;
|
|
|
|
RELEASE_SEM_LOCK(*sem);
|
|
|
|
retval = sem->id;
|
2002-07-09 16:24:59 +04:00
|
|
|
}
|
|
|
|
|
|
|
|
RELEASE_SEM_LIST_LOCK();
|
2002-07-25 05:05:51 +04:00
|
|
|
restore_interrupts(state);
|
2002-07-09 16:24:59 +04:00
|
|
|
|
2003-08-04 04:06:54 +04:00
|
|
|
if (!sem)
|
|
|
|
free(temp_name);
|
|
|
|
|
2002-07-09 16:24:59 +04:00
|
|
|
return retval;
|
|
|
|
}
|
|
|
|
|
2002-09-30 07:43:32 +04:00
|
|
|
|
|
|
|
sem_id
|
|
|
|
create_sem(int32 count, const char *name)
|
2002-07-09 16:24:59 +04:00
|
|
|
{
|
2002-08-03 04:41:27 +04:00
|
|
|
return create_sem_etc(count, name, team_get_kernel_team_id());
|
2002-07-09 16:24:59 +04:00
|
|
|
}
|
|
|
|
|
2002-09-30 07:43:32 +04:00
|
|
|
|
|
|
|
status_t
|
|
|
|
delete_sem(sem_id id)
|
2002-07-09 16:24:59 +04:00
|
|
|
{
|
2002-10-26 05:11:15 +04:00
|
|
|
return delete_sem_etc(id, 0, false);
|
2002-07-09 16:24:59 +04:00
|
|
|
}
|
|
|
|
|
2002-09-30 07:43:32 +04:00
|
|
|
|
|
|
|
status_t
|
2002-10-26 05:11:15 +04:00
|
|
|
delete_sem_etc(sem_id id, status_t return_code, bool interrupted)
|
2002-07-09 16:24:59 +04:00
|
|
|
{
|
|
|
|
int slot;
|
|
|
|
int state;
|
|
|
|
struct thread *t;
|
|
|
|
int released_threads;
|
|
|
|
char *old_name;
|
|
|
|
struct thread_queue release_queue;
|
|
|
|
|
2002-09-30 07:43:32 +04:00
|
|
|
if (gSemsActive == false)
|
2002-07-09 16:24:59 +04:00
|
|
|
return B_NO_MORE_SEMS;
|
2002-07-21 02:52:04 +04:00
|
|
|
if (id < 0)
|
2002-07-09 16:24:59 +04:00
|
|
|
return B_BAD_SEM_ID;
|
|
|
|
|
|
|
|
slot = id % MAX_SEMS;
|
|
|
|
|
2002-07-25 05:05:51 +04:00
|
|
|
state = disable_interrupts();
|
2002-09-30 07:43:32 +04:00
|
|
|
GRAB_SEM_LOCK(gSems[slot]);
|
2002-07-09 16:24:59 +04:00
|
|
|
|
2002-09-30 07:43:32 +04:00
|
|
|
if (gSems[slot].id != id) {
|
|
|
|
RELEASE_SEM_LOCK(gSems[slot]);
|
2002-07-25 05:05:51 +04:00
|
|
|
restore_interrupts(state);
|
2002-09-30 07:43:32 +04:00
|
|
|
TRACE(("delete_sem: invalid sem_id %ld\n", id));
|
2002-07-09 16:24:59 +04:00
|
|
|
return B_BAD_SEM_ID;
|
|
|
|
}
|
|
|
|
|
|
|
|
released_threads = 0;
|
|
|
|
release_queue.head = release_queue.tail = NULL;
|
|
|
|
|
|
|
|
// free any threads waiting for this semaphore
|
2003-08-04 04:06:54 +04:00
|
|
|
while ((t = thread_dequeue(&gSems[slot].u.used.q)) != NULL) {
|
2002-08-05 00:10:06 +04:00
|
|
|
t->state = B_THREAD_READY;
|
2002-10-26 05:11:15 +04:00
|
|
|
t->sem_errcode = interrupted ? B_INTERRUPTED : B_BAD_SEM_ID;
|
2002-07-09 16:24:59 +04:00
|
|
|
t->sem_deleted_retcode = return_code;
|
|
|
|
t->sem_count = 0;
|
|
|
|
thread_enqueue(t, &release_queue);
|
|
|
|
released_threads++;
|
|
|
|
}
|
|
|
|
|
2002-09-30 07:43:32 +04:00
|
|
|
gSems[slot].id = -1;
|
2003-08-04 04:06:54 +04:00
|
|
|
old_name = gSems[slot].u.used.name;
|
|
|
|
gSems[slot].u.used.name = NULL;
|
2002-07-09 16:24:59 +04:00
|
|
|
|
2002-09-30 07:43:32 +04:00
|
|
|
RELEASE_SEM_LOCK(gSems[slot]);
|
2002-07-09 16:24:59 +04:00
|
|
|
|
2003-08-04 04:06:54 +04:00
|
|
|
// append slot to the free list
|
|
|
|
GRAB_SEM_LIST_LOCK();
|
|
|
|
free_sem_slot(slot, id + MAX_SEMS);
|
|
|
|
RELEASE_SEM_LIST_LOCK();
|
|
|
|
|
2002-07-21 02:52:04 +04:00
|
|
|
if (released_threads > 0) {
|
2002-07-09 16:24:59 +04:00
|
|
|
GRAB_THREAD_LOCK();
|
2002-07-21 02:52:04 +04:00
|
|
|
while ((t = thread_dequeue(&release_queue)) != NULL) {
|
2003-01-27 06:05:09 +03:00
|
|
|
scheduler_enqueue_in_run_queue(t);
|
2002-07-09 16:24:59 +04:00
|
|
|
}
|
2003-01-27 06:05:09 +03:00
|
|
|
scheduler_reschedule();
|
2002-07-09 16:24:59 +04:00
|
|
|
RELEASE_THREAD_LOCK();
|
|
|
|
}
|
|
|
|
|
2002-07-25 05:05:51 +04:00
|
|
|
restore_interrupts(state);
|
2002-07-09 16:24:59 +04:00
|
|
|
|
2002-10-30 02:07:06 +03:00
|
|
|
free(old_name);
|
2002-07-09 16:24:59 +04:00
|
|
|
|
2002-09-30 07:43:32 +04:00
|
|
|
return B_OK;
|
2002-07-09 16:24:59 +04:00
|
|
|
}
|
|
|
|
|
2002-09-30 07:43:32 +04:00
|
|
|
|
|
|
|
/** Called from a timer handler. Wakes up a semaphore */
|
|
|
|
|
|
|
|
static int32
|
|
|
|
sem_timeout(timer *data)
|
2002-07-09 16:24:59 +04:00
|
|
|
{
|
2002-07-21 02:52:04 +04:00
|
|
|
struct sem_timeout_args *args = (struct sem_timeout_args *)data->entry.prev;
|
2002-07-09 16:24:59 +04:00
|
|
|
struct thread *t;
|
|
|
|
int slot;
|
|
|
|
int state;
|
|
|
|
struct thread_queue wakeup_queue;
|
|
|
|
|
|
|
|
t = thread_get_thread_struct(args->blocked_thread);
|
2002-07-21 02:52:04 +04:00
|
|
|
if (t == NULL)
|
2002-07-19 20:07:36 +04:00
|
|
|
return B_HANDLED_INTERRUPT;
|
2002-07-09 16:24:59 +04:00
|
|
|
slot = args->blocked_sem_id % MAX_SEMS;
|
|
|
|
|
2002-07-25 05:05:51 +04:00
|
|
|
state = disable_interrupts();
|
2002-09-30 07:43:32 +04:00
|
|
|
GRAB_SEM_LOCK(gSems[slot]);
|
2002-07-09 16:24:59 +04:00
|
|
|
|
2002-09-30 07:43:32 +04:00
|
|
|
TRACE(("sem_timeout: called on 0x%x sem %d, tid %d\n", to, to->sem_id, to->thread_id));
|
2002-07-09 16:24:59 +04:00
|
|
|
|
2002-09-30 07:43:32 +04:00
|
|
|
if (gSems[slot].id != args->blocked_sem_id) {
|
2002-07-09 16:24:59 +04:00
|
|
|
// this thread was not waiting on this semaphore
|
2002-09-03 06:09:48 +04:00
|
|
|
panic("sem_timeout: thid %ld was trying to wait on sem %ld which doesn't exist!\n",
|
2002-07-09 16:24:59 +04:00
|
|
|
args->blocked_thread, args->blocked_sem_id);
|
|
|
|
}
|
|
|
|
|
|
|
|
wakeup_queue.head = wakeup_queue.tail = NULL;
|
2002-09-30 07:43:32 +04:00
|
|
|
remove_thread_from_sem(t, &gSems[slot], &wakeup_queue, B_TIMED_OUT);
|
2002-10-31 16:20:00 +03:00
|
|
|
|
2002-09-30 07:43:32 +04:00
|
|
|
RELEASE_SEM_LOCK(gSems[slot]);
|
2002-07-09 16:24:59 +04:00
|
|
|
|
|
|
|
GRAB_THREAD_LOCK();
|
|
|
|
// put the threads in the run q here to make sure we dont deadlock in sem_interrupt_thread
|
2002-07-21 02:52:04 +04:00
|
|
|
while ((t = thread_dequeue(&wakeup_queue)) != NULL) {
|
2003-01-27 06:05:09 +03:00
|
|
|
scheduler_enqueue_in_run_queue(t);
|
2002-07-09 16:24:59 +04:00
|
|
|
}
|
|
|
|
RELEASE_THREAD_LOCK();
|
|
|
|
|
2002-07-25 05:05:51 +04:00
|
|
|
restore_interrupts(state);
|
2002-07-09 16:24:59 +04:00
|
|
|
|
2002-07-19 20:07:36 +04:00
|
|
|
return B_INVOKE_SCHEDULER;
|
2002-07-09 16:24:59 +04:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2002-09-30 07:43:32 +04:00
|
|
|
status_t
|
|
|
|
acquire_sem(sem_id id)
|
2002-07-09 16:24:59 +04:00
|
|
|
{
|
|
|
|
return acquire_sem_etc(id, 1, 0, 0);
|
|
|
|
}
|
|
|
|
|
2002-09-30 07:43:32 +04:00
|
|
|
|
|
|
|
status_t
|
|
|
|
acquire_sem_etc(sem_id id, int32 count, uint32 flags, bigtime_t timeout)
|
2002-07-09 16:24:59 +04:00
|
|
|
{
|
|
|
|
int slot = id % MAX_SEMS;
|
|
|
|
int state;
|
2002-09-30 07:43:32 +04:00
|
|
|
status_t status = B_OK;
|
2002-11-28 05:25:04 +03:00
|
|
|
|
2002-09-30 07:43:32 +04:00
|
|
|
if (gSemsActive == false)
|
2002-07-09 16:24:59 +04:00
|
|
|
return B_NO_MORE_SEMS;
|
2002-11-28 05:25:04 +03:00
|
|
|
|
|
|
|
if (!kernel_startup && !are_interrupts_enabled())
|
|
|
|
panic("acquire_sem_etc: called with interrupts disabled for sem %#lx\n", id);
|
|
|
|
|
2002-09-30 07:43:32 +04:00
|
|
|
if (id < 0)
|
2002-07-09 16:24:59 +04:00
|
|
|
return B_BAD_SEM_ID;
|
2002-07-21 02:52:04 +04:00
|
|
|
if (count <= 0)
|
2002-09-30 07:43:32 +04:00
|
|
|
return B_BAD_VALUE;
|
2002-07-09 16:24:59 +04:00
|
|
|
|
2002-07-25 05:05:51 +04:00
|
|
|
state = disable_interrupts();
|
2002-09-30 07:43:32 +04:00
|
|
|
GRAB_SEM_LOCK(gSems[slot]);
|
2002-07-21 02:52:04 +04:00
|
|
|
|
2002-09-30 07:43:32 +04:00
|
|
|
if (gSems[slot].id != id) {
|
|
|
|
TRACE(("acquire_sem_etc: bad sem_id %ld\n", id));
|
|
|
|
status = B_BAD_SEM_ID;
|
2002-07-09 16:24:59 +04:00
|
|
|
goto err;
|
|
|
|
}
|
|
|
|
|
2003-08-04 04:06:54 +04:00
|
|
|
if (gSems[slot].u.used.count - count < 0 && (flags & B_TIMEOUT) != 0
|
|
|
|
&& timeout <= 0) {
|
2002-07-09 16:24:59 +04:00
|
|
|
// immediate timeout
|
2003-08-04 02:21:00 +04:00
|
|
|
status = B_WOULD_BLOCK;
|
2002-07-09 16:24:59 +04:00
|
|
|
goto err;
|
|
|
|
}
|
|
|
|
|
2003-08-04 04:06:54 +04:00
|
|
|
if ((gSems[slot].u.used.count -= count) < 0) {
|
2002-07-09 16:24:59 +04:00
|
|
|
// we need to block
|
|
|
|
struct thread *t = thread_get_current_thread();
|
2002-07-21 02:52:04 +04:00
|
|
|
timer timeout_timer; // stick it on the stack, since we may be blocking here
|
2002-07-09 16:24:59 +04:00
|
|
|
struct sem_timeout_args args;
|
|
|
|
|
2003-08-04 04:06:54 +04:00
|
|
|
TRACE_BLOCK(("acquire_sem_etc(id = %ld): block name = %s, thread = %p,"
|
|
|
|
" name = %s\n", id, gSems[slot].u.used.name, t, t->name));
|
2002-09-30 07:43:32 +04:00
|
|
|
|
2002-10-23 21:31:10 +04:00
|
|
|
// do a quick check to see if the thread has any pending signals
|
2002-07-09 16:24:59 +04:00
|
|
|
// this should catch most of the cases where the thread had a signal
|
2002-10-23 21:31:10 +04:00
|
|
|
if ((flags & B_CAN_INTERRUPT) && t->sig_pending) {
|
2003-08-04 04:06:54 +04:00
|
|
|
gSems[slot].u.used.count += count;
|
2002-09-30 07:43:32 +04:00
|
|
|
status = B_INTERRUPTED;
|
2002-07-09 16:24:59 +04:00
|
|
|
goto err;
|
|
|
|
}
|
|
|
|
|
2002-08-05 00:10:06 +04:00
|
|
|
t->next_state = B_THREAD_WAITING;
|
2002-07-09 16:24:59 +04:00
|
|
|
t->sem_flags = flags;
|
|
|
|
t->sem_blocking = id;
|
|
|
|
t->sem_acquire_count = count;
|
2003-08-04 04:06:54 +04:00
|
|
|
t->sem_count = min(-gSems[slot].u.used.count, count);
|
|
|
|
// store the count we need to restore upon release
|
2002-07-09 16:24:59 +04:00
|
|
|
t->sem_deleted_retcode = 0;
|
|
|
|
t->sem_errcode = B_NO_ERROR;
|
2003-08-04 04:06:54 +04:00
|
|
|
thread_enqueue(t, &gSems[slot].u.used.q);
|
2002-07-09 16:24:59 +04:00
|
|
|
|
2002-07-21 02:52:04 +04:00
|
|
|
if ((flags & (B_TIMEOUT | B_ABSOLUTE_TIMEOUT)) != 0) {
|
2002-09-30 07:43:32 +04:00
|
|
|
TRACE(("sem_acquire_etc: setting timeout sem for %d %d usecs, semid %d, tid %d\n",
|
|
|
|
timeout, sem_id, t->id));
|
|
|
|
|
2002-07-09 16:24:59 +04:00
|
|
|
// set up an event to go off with the thread struct as the data
|
|
|
|
args.blocked_sem_id = id;
|
|
|
|
args.blocked_thread = t->id;
|
|
|
|
args.sem_count = count;
|
|
|
|
|
2002-07-21 02:52:04 +04:00
|
|
|
// another evil hack: pass the args into timer->entry.prev
|
|
|
|
timeout_timer.entry.prev = (qent *)&args;
|
|
|
|
add_timer(&timeout_timer, &sem_timeout, timeout,
|
|
|
|
flags & B_RELATIVE_TIMEOUT ?
|
|
|
|
B_ONE_SHOT_RELATIVE_TIMER : B_ONE_SHOT_ABSOLUTE_TIMER);
|
2002-07-09 16:24:59 +04:00
|
|
|
}
|
|
|
|
|
2002-09-30 07:43:32 +04:00
|
|
|
RELEASE_SEM_LOCK(gSems[slot]);
|
2002-07-09 16:24:59 +04:00
|
|
|
GRAB_THREAD_LOCK();
|
2002-10-23 21:31:10 +04:00
|
|
|
// check again to see if a signal is pending.
|
2002-07-09 16:24:59 +04:00
|
|
|
// it may have been delivered while setting up the sem, though it's pretty unlikely
|
2002-10-23 21:31:10 +04:00
|
|
|
if ((flags & B_CAN_INTERRUPT) && t->sig_pending) {
|
2002-07-09 16:24:59 +04:00
|
|
|
struct thread_queue wakeup_queue;
|
|
|
|
// ok, so a tiny race happened where a signal was delivered to this thread while
|
|
|
|
// it was setting up the sem. We can only be sure a signal wasn't delivered
|
|
|
|
// here, since the threadlock is held. The previous check would have found most
|
|
|
|
// instances, but there was a race, so we have to handle it. It'll be more messy...
|
|
|
|
wakeup_queue.head = wakeup_queue.tail = NULL;
|
2002-09-30 07:43:32 +04:00
|
|
|
GRAB_SEM_LOCK(gSems[slot]);
|
|
|
|
if (gSems[slot].id == id) {
|
|
|
|
remove_thread_from_sem(t, &gSems[slot], &wakeup_queue, EINTR);
|
2002-07-09 16:24:59 +04:00
|
|
|
}
|
2002-09-30 07:43:32 +04:00
|
|
|
RELEASE_SEM_LOCK(gSems[slot]);
|
2002-07-21 02:52:04 +04:00
|
|
|
while ((t = thread_dequeue(&wakeup_queue)) != NULL) {
|
2003-01-27 06:05:09 +03:00
|
|
|
scheduler_enqueue_in_run_queue(t);
|
2002-07-09 16:24:59 +04:00
|
|
|
}
|
|
|
|
// fall through and reschedule since another thread with a higher priority may have been woken up
|
|
|
|
}
|
2003-01-27 06:05:09 +03:00
|
|
|
scheduler_reschedule();
|
2002-07-09 16:24:59 +04:00
|
|
|
RELEASE_THREAD_LOCK();
|
|
|
|
|
2002-07-21 02:52:04 +04:00
|
|
|
if ((flags & (B_TIMEOUT | B_ABSOLUTE_TIMEOUT)) != 0) {
|
|
|
|
if (t->sem_errcode != B_TIMED_OUT) {
|
2002-07-09 16:24:59 +04:00
|
|
|
// cancel the timer event, the sem may have been deleted or interrupted
|
|
|
|
// with the timer still active
|
2002-07-21 02:52:04 +04:00
|
|
|
cancel_timer(&timeout_timer);
|
2002-07-09 16:24:59 +04:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2002-07-25 05:05:51 +04:00
|
|
|
restore_interrupts(state);
|
2002-07-09 16:24:59 +04:00
|
|
|
|
2003-08-04 04:06:54 +04:00
|
|
|
TRACE_BLOCK(("acquire_sem_etc(id = %ld): exit block name = %s, "
|
|
|
|
"thread = %p (%s)\n", id, gSems[slot].u.used.name, t,
|
|
|
|
t->name));
|
2002-07-09 16:24:59 +04:00
|
|
|
return t->sem_errcode;
|
|
|
|
}
|
|
|
|
|
|
|
|
err:
|
2002-09-30 07:43:32 +04:00
|
|
|
RELEASE_SEM_LOCK(gSems[slot]);
|
2002-07-25 05:05:51 +04:00
|
|
|
restore_interrupts(state);
|
2002-07-09 16:24:59 +04:00
|
|
|
|
2002-09-30 07:43:32 +04:00
|
|
|
return status;
|
2002-07-09 16:24:59 +04:00
|
|
|
}
|
|
|
|
|
2002-09-30 07:43:32 +04:00
|
|
|
|
|
|
|
status_t
|
|
|
|
release_sem(sem_id id)
|
2002-07-09 16:24:59 +04:00
|
|
|
{
|
|
|
|
return release_sem_etc(id, 1, 0);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2002-09-30 07:43:32 +04:00
|
|
|
status_t
|
|
|
|
release_sem_etc(sem_id id, int32 count, uint32 flags)
|
2002-07-09 16:24:59 +04:00
|
|
|
{
|
|
|
|
int slot = id % MAX_SEMS;
|
|
|
|
int state;
|
|
|
|
int released_threads = 0;
|
|
|
|
struct thread_queue release_queue;
|
2002-09-30 07:43:32 +04:00
|
|
|
status_t status = B_OK;
|
2002-07-09 16:24:59 +04:00
|
|
|
|
2002-09-30 07:43:32 +04:00
|
|
|
if (gSemsActive == false)
|
2002-07-09 16:24:59 +04:00
|
|
|
return B_NO_MORE_SEMS;
|
2002-07-21 02:52:04 +04:00
|
|
|
if (id < 0)
|
2002-07-09 16:24:59 +04:00
|
|
|
return B_BAD_SEM_ID;
|
2002-07-21 02:52:04 +04:00
|
|
|
if (count <= 0)
|
2002-09-30 07:43:32 +04:00
|
|
|
return B_BAD_VALUE;
|
2002-07-09 16:24:59 +04:00
|
|
|
|
2002-07-25 05:05:51 +04:00
|
|
|
state = disable_interrupts();
|
2002-09-30 07:43:32 +04:00
|
|
|
GRAB_SEM_LOCK(gSems[slot]);
|
2002-07-09 16:24:59 +04:00
|
|
|
|
2002-09-30 07:43:32 +04:00
|
|
|
if (gSems[slot].id != id) {
|
|
|
|
TRACE(("sem_release_etc: invalid sem_id %ld\n", id));
|
|
|
|
status = B_BAD_SEM_ID;
|
2002-07-09 16:24:59 +04:00
|
|
|
goto err;
|
|
|
|
}
|
|
|
|
|
|
|
|
// clear out a queue we will use to hold all of the threads that we will have to
|
|
|
|
// put back into the run list. This is done so the thread lock wont be held
|
|
|
|
// while this sems lock is held since the two locks are grabbed in the other
|
|
|
|
// order in sem_interrupt_thread.
|
|
|
|
release_queue.head = release_queue.tail = NULL;
|
|
|
|
|
2002-07-21 02:52:04 +04:00
|
|
|
while (count > 0) {
|
2002-07-09 16:24:59 +04:00
|
|
|
int delta = count;
|
2003-08-04 04:06:54 +04:00
|
|
|
if (gSems[slot].u.used.count < 0) {
|
|
|
|
struct thread *t = thread_lookat_queue(&gSems[slot].u.used.q);
|
2002-07-09 16:24:59 +04:00
|
|
|
|
|
|
|
delta = min(count, t->sem_count);
|
|
|
|
t->sem_count -= delta;
|
2002-07-21 02:52:04 +04:00
|
|
|
if (t->sem_count <= 0) {
|
2002-07-09 16:24:59 +04:00
|
|
|
// release this thread
|
2003-08-04 04:06:54 +04:00
|
|
|
t = thread_dequeue(&gSems[slot].u.used.q);
|
2002-07-09 16:24:59 +04:00
|
|
|
thread_enqueue(t, &release_queue);
|
2002-08-05 00:10:06 +04:00
|
|
|
t->state = B_THREAD_READY;
|
2002-07-09 16:24:59 +04:00
|
|
|
released_threads++;
|
|
|
|
t->sem_count = 0;
|
|
|
|
t->sem_deleted_retcode = 0;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2003-08-04 04:06:54 +04:00
|
|
|
gSems[slot].u.used.count += delta;
|
2002-07-09 16:24:59 +04:00
|
|
|
count -= delta;
|
|
|
|
}
|
2002-09-30 07:43:32 +04:00
|
|
|
RELEASE_SEM_LOCK(gSems[slot]);
|
2002-07-09 16:24:59 +04:00
|
|
|
|
|
|
|
// pull off any items in the release queue and put them in the run queue
|
2002-07-21 02:52:04 +04:00
|
|
|
if (released_threads > 0) {
|
2002-07-09 16:24:59 +04:00
|
|
|
struct thread *t;
|
2002-08-01 00:05:24 +04:00
|
|
|
int priority;
|
2002-07-09 16:24:59 +04:00
|
|
|
GRAB_THREAD_LOCK();
|
2002-07-21 02:52:04 +04:00
|
|
|
while ((t = thread_dequeue(&release_queue)) != NULL) {
|
2002-08-01 00:05:24 +04:00
|
|
|
// temporarily place thread in a run queue with high priority to boost it up
|
|
|
|
priority = t->priority;
|
2002-08-05 00:10:06 +04:00
|
|
|
t->priority = t->priority >= B_FIRST_REAL_TIME_PRIORITY ? t->priority : B_FIRST_REAL_TIME_PRIORITY;
|
2003-01-27 06:05:09 +03:00
|
|
|
scheduler_enqueue_in_run_queue(t);
|
2002-08-01 00:05:24 +04:00
|
|
|
t->priority = priority;
|
2002-07-09 16:24:59 +04:00
|
|
|
}
|
2003-01-27 06:05:09 +03:00
|
|
|
if ((flags & B_DO_NOT_RESCHEDULE) == 0)
|
|
|
|
scheduler_reschedule();
|
|
|
|
|
2002-07-09 16:24:59 +04:00
|
|
|
RELEASE_THREAD_LOCK();
|
|
|
|
}
|
|
|
|
goto outnolock;
|
|
|
|
|
|
|
|
err:
|
2002-09-30 07:43:32 +04:00
|
|
|
RELEASE_SEM_LOCK(gSems[slot]);
|
2002-07-09 16:24:59 +04:00
|
|
|
outnolock:
|
2002-07-25 05:05:51 +04:00
|
|
|
restore_interrupts(state);
|
2002-07-09 16:24:59 +04:00
|
|
|
|
2002-09-30 07:43:32 +04:00
|
|
|
return status;
|
2002-07-09 16:24:59 +04:00
|
|
|
}
|
|
|
|
|
2002-09-30 07:43:32 +04:00
|
|
|
|
|
|
|
status_t
|
|
|
|
get_sem_count(sem_id id, int32 *thread_count)
|
2002-07-09 16:24:59 +04:00
|
|
|
{
|
|
|
|
int slot;
|
|
|
|
int state;
|
|
|
|
|
2002-09-30 07:43:32 +04:00
|
|
|
if (gSemsActive == false)
|
2002-07-09 16:24:59 +04:00
|
|
|
return B_NO_MORE_SEMS;
|
2002-07-21 02:52:04 +04:00
|
|
|
if (id < 0)
|
2002-07-09 16:24:59 +04:00
|
|
|
return B_BAD_SEM_ID;
|
|
|
|
if (thread_count == NULL)
|
2002-07-14 14:14:29 +04:00
|
|
|
return EINVAL;
|
2002-07-09 16:24:59 +04:00
|
|
|
|
|
|
|
slot = id % MAX_SEMS;
|
|
|
|
|
2002-07-25 05:05:51 +04:00
|
|
|
state = disable_interrupts();
|
2002-09-30 07:43:32 +04:00
|
|
|
GRAB_SEM_LOCK(gSems[slot]);
|
2002-07-09 16:24:59 +04:00
|
|
|
|
2002-09-30 07:43:32 +04:00
|
|
|
if (gSems[slot].id != id) {
|
|
|
|
RELEASE_SEM_LOCK(gSems[slot]);
|
2002-07-25 05:05:51 +04:00
|
|
|
restore_interrupts(state);
|
2002-09-30 07:43:32 +04:00
|
|
|
TRACE(("sem_get_count: invalid sem_id %ld\n", id));
|
2002-07-09 16:24:59 +04:00
|
|
|
return B_BAD_SEM_ID;
|
|
|
|
}
|
|
|
|
|
2003-08-04 04:06:54 +04:00
|
|
|
*thread_count = gSems[slot].u.used.count;
|
2002-07-09 16:24:59 +04:00
|
|
|
|
2002-09-30 07:43:32 +04:00
|
|
|
RELEASE_SEM_LOCK(gSems[slot]);
|
2002-07-25 05:05:51 +04:00
|
|
|
restore_interrupts(state);
|
2002-07-09 16:24:59 +04:00
|
|
|
|
|
|
|
return B_NO_ERROR;
|
|
|
|
}
|
|
|
|
|
2002-09-30 07:43:32 +04:00
|
|
|
|
2004-02-23 07:26:44 +03:00
|
|
|
/** Fills the thread_info structure with information from the specified
|
|
|
|
* thread.
|
|
|
|
* The thread lock must be held when called.
|
|
|
|
*/
|
|
|
|
|
|
|
|
static void
|
|
|
|
fill_sem_info(struct sem_entry *sem, sem_info *info, size_t size)
|
|
|
|
{
|
|
|
|
info->sem = sem->id;
|
|
|
|
info->team = sem->u.used.owner;
|
|
|
|
strlcpy(info->name, sem->u.used.name, sizeof(info->name));
|
|
|
|
info->count = sem->u.used.count;
|
|
|
|
info->latest_holder = sem->u.used.q.head->id;
|
|
|
|
// ToDo: not sure if this is the latest holder, or the next
|
|
|
|
// holder...
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2002-09-30 07:43:32 +04:00
|
|
|
/** The underscore is needed for binary compatibility with BeOS.
|
|
|
|
* OS.h contains the following macro:
|
|
|
|
* #define get_sem_info(sem, info) \
|
|
|
|
* _get_sem_info((sem), (info), sizeof(*(info)))
|
|
|
|
*/
|
|
|
|
|
|
|
|
status_t
|
2004-02-23 07:26:44 +03:00
|
|
|
_get_sem_info(sem_id id, struct sem_info *info, size_t size)
|
2002-07-09 16:24:59 +04:00
|
|
|
{
|
2004-02-23 07:26:44 +03:00
|
|
|
status_t status = B_OK;
|
2002-07-09 16:24:59 +04:00
|
|
|
int state;
|
|
|
|
int slot;
|
|
|
|
|
2004-02-23 07:26:44 +03:00
|
|
|
if (!gSemsActive)
|
2002-07-09 16:24:59 +04:00
|
|
|
return B_NO_MORE_SEMS;
|
2002-07-21 02:52:04 +04:00
|
|
|
if (id < 0)
|
2002-07-09 16:24:59 +04:00
|
|
|
return B_BAD_SEM_ID;
|
2004-02-23 07:26:44 +03:00
|
|
|
if (info == NULL || size != sizeof(sem_info))
|
|
|
|
return B_BAD_VALUE;
|
2002-07-09 16:24:59 +04:00
|
|
|
|
|
|
|
slot = id % MAX_SEMS;
|
|
|
|
|
2002-07-25 05:05:51 +04:00
|
|
|
state = disable_interrupts();
|
2002-09-30 07:43:32 +04:00
|
|
|
GRAB_SEM_LOCK(gSems[slot]);
|
2002-07-09 16:24:59 +04:00
|
|
|
|
2002-09-30 07:43:32 +04:00
|
|
|
if (gSems[slot].id != id) {
|
2004-02-23 07:26:44 +03:00
|
|
|
status = B_BAD_SEM_ID;
|
2002-09-30 07:43:32 +04:00
|
|
|
TRACE(("get_sem_info: invalid sem_id %ld\n", id));
|
2004-02-23 07:26:44 +03:00
|
|
|
} else
|
|
|
|
fill_sem_info(&gSems[slot], info, size);
|
2002-07-09 16:24:59 +04:00
|
|
|
|
2002-09-30 07:43:32 +04:00
|
|
|
RELEASE_SEM_LOCK(gSems[slot]);
|
2002-07-25 05:05:51 +04:00
|
|
|
restore_interrupts(state);
|
2002-07-09 16:24:59 +04:00
|
|
|
|
2004-02-23 07:26:44 +03:00
|
|
|
return status;
|
2002-07-09 16:24:59 +04:00
|
|
|
}
|
|
|
|
|
2002-09-30 07:43:32 +04:00
|
|
|
|
|
|
|
/** The underscore is needed for binary compatibility with BeOS.
|
|
|
|
* OS.h contains the following macro:
|
|
|
|
* #define get_next_sem_info(team, cookie, info) \
|
|
|
|
* _get_next_sem_info((team), (cookie), (info), sizeof(*(info)))
|
|
|
|
*/
|
|
|
|
|
|
|
|
status_t
|
2004-02-23 07:26:44 +03:00
|
|
|
_get_next_sem_info(team_id team, int32 *_cookie, struct sem_info *info, size_t size)
|
2002-07-09 16:24:59 +04:00
|
|
|
{
|
|
|
|
int state;
|
|
|
|
int slot;
|
2003-08-04 02:21:00 +04:00
|
|
|
bool found = false;
|
2002-07-09 16:24:59 +04:00
|
|
|
|
2004-02-23 07:26:44 +03:00
|
|
|
if (!gSemsActive)
|
2002-07-09 16:24:59 +04:00
|
|
|
return B_NO_MORE_SEMS;
|
2004-02-23 07:26:44 +03:00
|
|
|
if (_cookie == NULL || info == NULL || size != sizeof(sem_info))
|
|
|
|
return B_BAD_VALUE;
|
|
|
|
|
|
|
|
if (team == B_CURRENT_TEAM)
|
|
|
|
team = team_get_current_team_id();
|
2002-09-30 07:43:32 +04:00
|
|
|
/* prevents gSems[].owner == -1 >= means owned by a port */
|
2004-02-23 07:26:44 +03:00
|
|
|
if (team < 0 || !team_is_valid(team))
|
2002-09-30 07:43:32 +04:00
|
|
|
return B_BAD_TEAM_ID;
|
2002-07-09 16:24:59 +04:00
|
|
|
|
2004-02-23 07:26:44 +03:00
|
|
|
slot = *_cookie;
|
|
|
|
if (slot >= MAX_SEMS)
|
|
|
|
return B_BAD_VALUE;
|
|
|
|
|
2002-07-25 05:05:51 +04:00
|
|
|
state = disable_interrupts();
|
2002-07-09 16:24:59 +04:00
|
|
|
GRAB_SEM_LIST_LOCK();
|
|
|
|
|
|
|
|
while (slot < MAX_SEMS) {
|
2003-08-04 04:06:54 +04:00
|
|
|
if (gSems[slot].id != -1 && gSems[slot].u.used.owner == team) {
|
2002-09-30 07:43:32 +04:00
|
|
|
GRAB_SEM_LOCK(gSems[slot]);
|
2003-08-04 04:06:54 +04:00
|
|
|
if (gSems[slot].id != -1 && gSems[slot].u.used.owner == team) {
|
2002-07-09 16:24:59 +04:00
|
|
|
// found one!
|
2004-02-23 07:26:44 +03:00
|
|
|
fill_sem_info(&gSems[slot], info, size);
|
2002-07-09 16:24:59 +04:00
|
|
|
|
2002-09-30 07:43:32 +04:00
|
|
|
RELEASE_SEM_LOCK(gSems[slot]);
|
2002-07-09 16:24:59 +04:00
|
|
|
slot++;
|
2003-08-04 02:21:00 +04:00
|
|
|
found = true;
|
2002-07-09 16:24:59 +04:00
|
|
|
break;
|
|
|
|
}
|
2002-09-30 07:43:32 +04:00
|
|
|
RELEASE_SEM_LOCK(gSems[slot]);
|
|
|
|
}
|
2002-07-09 16:24:59 +04:00
|
|
|
slot++;
|
|
|
|
}
|
|
|
|
RELEASE_SEM_LIST_LOCK();
|
2002-07-25 05:05:51 +04:00
|
|
|
restore_interrupts(state);
|
2002-07-09 16:24:59 +04:00
|
|
|
|
2003-08-04 02:21:00 +04:00
|
|
|
if (!found)
|
|
|
|
return B_BAD_VALUE;
|
2004-02-23 07:26:44 +03:00
|
|
|
|
|
|
|
*_cookie = slot;
|
|
|
|
return B_OK;
|
2002-07-09 16:24:59 +04:00
|
|
|
}
|
|
|
|
|
2002-09-30 07:43:32 +04:00
|
|
|
|
|
|
|
status_t
|
|
|
|
set_sem_owner(sem_id id, team_id team)
|
2002-07-09 16:24:59 +04:00
|
|
|
{
|
|
|
|
int state;
|
|
|
|
int slot;
|
|
|
|
|
2002-09-30 07:43:32 +04:00
|
|
|
if (gSemsActive == false)
|
2002-07-09 16:24:59 +04:00
|
|
|
return B_NO_MORE_SEMS;
|
2002-07-21 02:52:04 +04:00
|
|
|
if (id < 0)
|
2002-07-09 16:24:59 +04:00
|
|
|
return B_BAD_SEM_ID;
|
2004-02-23 06:31:08 +03:00
|
|
|
if (team < 0 || !team_is_valid(team))
|
2002-09-30 07:43:32 +04:00
|
|
|
return B_BAD_TEAM_ID;
|
2002-07-09 16:24:59 +04:00
|
|
|
|
|
|
|
slot = id % MAX_SEMS;
|
|
|
|
|
2002-07-25 05:05:51 +04:00
|
|
|
state = disable_interrupts();
|
2002-09-30 07:43:32 +04:00
|
|
|
GRAB_SEM_LOCK(gSems[slot]);
|
2002-07-09 16:24:59 +04:00
|
|
|
|
2002-09-30 07:43:32 +04:00
|
|
|
if (gSems[slot].id != id) {
|
|
|
|
RELEASE_SEM_LOCK(gSems[slot]);
|
2002-07-25 05:05:51 +04:00
|
|
|
restore_interrupts(state);
|
2002-09-30 07:43:32 +04:00
|
|
|
TRACE(("set_sem_owner: invalid sem_id %ld\n", id));
|
2002-07-09 16:24:59 +04:00
|
|
|
return B_BAD_SEM_ID;
|
|
|
|
}
|
|
|
|
|
2004-02-23 06:31:08 +03:00
|
|
|
// ToDo: this is a small race condition: the team ID could already
|
2002-09-30 07:48:59 +04:00
|
|
|
// be invalid at this point - we would lose one semaphore slot in
|
|
|
|
// this case!
|
2004-02-23 06:31:08 +03:00
|
|
|
// The only safe way to do this is to prevent either team (the new
|
|
|
|
// or the old owner) from dying until we leave the spinlock.
|
2003-08-04 04:06:54 +04:00
|
|
|
gSems[slot].u.used.owner = team;
|
2002-07-09 16:24:59 +04:00
|
|
|
|
2002-09-30 07:43:32 +04:00
|
|
|
RELEASE_SEM_LOCK(gSems[slot]);
|
2002-07-25 05:05:51 +04:00
|
|
|
restore_interrupts(state);
|
2002-07-09 16:24:59 +04:00
|
|
|
|
|
|
|
return B_NO_ERROR;
|
|
|
|
}
|
|
|
|
|
2002-09-30 07:43:32 +04:00
|
|
|
|
|
|
|
/** Wake up a thread that's blocked on a semaphore
|
|
|
|
* this function must be entered with interrupts disabled and THREADLOCK held
|
|
|
|
*/
|
|
|
|
|
|
|
|
status_t
|
|
|
|
sem_interrupt_thread(struct thread *t)
|
2002-07-09 16:24:59 +04:00
|
|
|
{
|
|
|
|
int slot;
|
|
|
|
struct thread_queue wakeup_queue;
|
|
|
|
|
2004-02-23 06:31:08 +03:00
|
|
|
TRACE(("sem_interrupt_thread: called on thread %p (%d), blocked on sem 0x%x\n",
|
|
|
|
t, t->id, t->sem_blocking));
|
2002-07-09 16:24:59 +04:00
|
|
|
|
2002-08-05 00:10:06 +04:00
|
|
|
if (t->state != B_THREAD_WAITING || t->sem_blocking < 0)
|
2004-02-23 06:31:08 +03:00
|
|
|
return B_BAD_VALUE;
|
2002-10-31 16:20:00 +03:00
|
|
|
if (!(t->sem_flags & B_CAN_INTERRUPT))
|
2004-02-23 06:31:08 +03:00
|
|
|
return B_NOT_ALLOWED;
|
2002-07-09 16:24:59 +04:00
|
|
|
|
|
|
|
slot = t->sem_blocking % MAX_SEMS;
|
|
|
|
|
2002-09-30 07:43:32 +04:00
|
|
|
GRAB_SEM_LOCK(gSems[slot]);
|
2002-07-09 16:24:59 +04:00
|
|
|
|
2002-09-30 07:43:32 +04:00
|
|
|
if (gSems[slot].id != t->sem_blocking) {
|
2002-09-03 06:09:48 +04:00
|
|
|
panic("sem_interrupt_thread: thread 0x%lx sez it's blocking on sem 0x%lx, but that sem doesn't exist!\n", t->id, t->sem_blocking);
|
2002-07-09 16:24:59 +04:00
|
|
|
}
|
|
|
|
|
|
|
|
wakeup_queue.head = wakeup_queue.tail = NULL;
|
2004-02-23 06:31:08 +03:00
|
|
|
if (remove_thread_from_sem(t, &gSems[slot], &wakeup_queue, EINTR) != B_OK)
|
2002-09-03 06:09:48 +04:00
|
|
|
panic("sem_interrupt_thread: thread 0x%lx not found in sem 0x%lx's wait queue\n", t->id, t->sem_blocking);
|
2002-07-09 16:24:59 +04:00
|
|
|
|
2002-09-30 07:43:32 +04:00
|
|
|
RELEASE_SEM_LOCK(gSems[slot]);
|
2002-07-09 16:24:59 +04:00
|
|
|
|
2002-07-21 02:52:04 +04:00
|
|
|
while ((t = thread_dequeue(&wakeup_queue)) != NULL) {
|
2003-01-27 06:05:09 +03:00
|
|
|
scheduler_enqueue_in_run_queue(t);
|
2002-07-09 16:24:59 +04:00
|
|
|
}
|
|
|
|
|
|
|
|
return B_NO_ERROR;
|
|
|
|
}
|
|
|
|
|
2002-09-30 07:43:32 +04:00
|
|
|
|
|
|
|
/** forcibly removes a thread from a semaphores wait q. May have to wake up other threads in the
|
|
|
|
* process. All threads that need to be woken up are added to the passed in thread_queue.
|
|
|
|
* must be called with sem lock held
|
|
|
|
*/
|
|
|
|
|
|
|
|
static int
|
|
|
|
remove_thread_from_sem(struct thread *t, struct sem_entry *sem, struct thread_queue *queue, int sem_errcode)
|
2002-07-09 16:24:59 +04:00
|
|
|
{
|
|
|
|
struct thread *t1;
|
|
|
|
|
|
|
|
// remove the thread from the queue and place it in the supplied queue
|
2003-08-04 04:06:54 +04:00
|
|
|
t1 = thread_dequeue_id(&sem->u.used.q, t->id);
|
2002-09-30 07:48:59 +04:00
|
|
|
if (t != t1)
|
2004-02-23 06:31:08 +03:00
|
|
|
return B_ENTRY_NOT_FOUND;
|
|
|
|
|
2003-08-04 04:06:54 +04:00
|
|
|
sem->u.used.count += t->sem_acquire_count;
|
2002-10-31 16:20:00 +03:00
|
|
|
t->state = t->next_state = B_THREAD_READY;
|
2002-07-09 16:24:59 +04:00
|
|
|
t->sem_errcode = sem_errcode;
|
|
|
|
thread_enqueue(t, queue);
|
|
|
|
|
|
|
|
// now see if more threads need to be woken up
|
2003-08-04 04:06:54 +04:00
|
|
|
while (sem->u.used.count > 0
|
|
|
|
&& (t1 = thread_lookat_queue(&sem->u.used.q))) {
|
|
|
|
int delta = min(t->sem_count, sem->u.used.count);
|
2002-07-09 16:24:59 +04:00
|
|
|
|
|
|
|
t->sem_count -= delta;
|
2002-09-30 07:48:59 +04:00
|
|
|
if (t->sem_count <= 0) {
|
2003-08-04 04:06:54 +04:00
|
|
|
t = thread_dequeue(&sem->u.used.q);
|
2002-10-31 17:32:56 +03:00
|
|
|
t->state = t->next_state = B_THREAD_READY;
|
2002-07-09 16:24:59 +04:00
|
|
|
thread_enqueue(t, queue);
|
|
|
|
}
|
2003-08-04 04:06:54 +04:00
|
|
|
sem->u.used.count -= delta;
|
2002-07-09 16:24:59 +04:00
|
|
|
}
|
|
|
|
return B_NO_ERROR;
|
|
|
|
}
|
|
|
|
|
2002-09-30 07:43:32 +04:00
|
|
|
|
|
|
|
/** this function cycles through the sem table, deleting all the sems that are owned by
|
|
|
|
* the passed team_id
|
|
|
|
*/
|
|
|
|
|
|
|
|
int
|
|
|
|
sem_delete_owned_sems(team_id owner)
|
2002-07-09 16:24:59 +04:00
|
|
|
{
|
|
|
|
int state;
|
|
|
|
int i;
|
|
|
|
int count = 0;
|
|
|
|
|
|
|
|
if (owner < 0)
|
2002-09-30 07:43:32 +04:00
|
|
|
return B_BAD_TEAM_ID;
|
2002-07-09 16:24:59 +04:00
|
|
|
|
2002-07-25 05:05:51 +04:00
|
|
|
state = disable_interrupts();
|
2002-07-09 16:24:59 +04:00
|
|
|
GRAB_SEM_LIST_LOCK();
|
|
|
|
|
2002-09-30 07:43:32 +04:00
|
|
|
for (i = 0; i < MAX_SEMS; i++) {
|
2003-08-04 04:06:54 +04:00
|
|
|
if (gSems[i].id != -1 && gSems[i].u.used.owner == owner) {
|
2002-09-30 07:43:32 +04:00
|
|
|
sem_id id = gSems[i].id;
|
2002-07-09 16:24:59 +04:00
|
|
|
|
|
|
|
RELEASE_SEM_LIST_LOCK();
|
2002-07-25 05:05:51 +04:00
|
|
|
restore_interrupts(state);
|
2002-07-09 16:24:59 +04:00
|
|
|
|
2002-10-26 05:11:15 +04:00
|
|
|
delete_sem(id);
|
2002-07-09 16:24:59 +04:00
|
|
|
count++;
|
|
|
|
|
2002-07-25 05:05:51 +04:00
|
|
|
state = disable_interrupts();
|
2002-07-09 16:24:59 +04:00
|
|
|
GRAB_SEM_LIST_LOCK();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
RELEASE_SEM_LIST_LOCK();
|
2002-07-25 05:05:51 +04:00
|
|
|
restore_interrupts(state);
|
2002-07-09 16:24:59 +04:00
|
|
|
|
|
|
|
return count;
|
|
|
|
}
|
|
|
|
|
2002-09-30 07:43:32 +04:00
|
|
|
|
2004-02-23 06:54:00 +03:00
|
|
|
// #pragma mark -
|
|
|
|
|
|
|
|
|
2002-09-30 07:43:32 +04:00
|
|
|
sem_id
|
2004-02-23 06:54:00 +03:00
|
|
|
user_create_sem(int32 count, const char *userName)
|
2002-07-09 16:24:59 +04:00
|
|
|
{
|
2004-02-23 06:54:00 +03:00
|
|
|
char name[B_OS_NAME_LENGTH];
|
2002-07-09 16:24:59 +04:00
|
|
|
|
2004-02-23 06:54:00 +03:00
|
|
|
if (userName == NULL)
|
|
|
|
return create_sem_etc(count, NULL, team_get_current_team_id());
|
2002-07-09 16:24:59 +04:00
|
|
|
|
2004-02-23 06:54:00 +03:00
|
|
|
if (!IS_USER_ADDRESS(userName)
|
|
|
|
|| user_strlcpy(name, userName, B_OS_NAME_LENGTH) < B_OK)
|
|
|
|
return B_BAD_ADDRESS;
|
2002-09-30 07:43:32 +04:00
|
|
|
|
2004-02-23 06:54:00 +03:00
|
|
|
return create_sem_etc(count, name, team_get_current_team_id());
|
2002-07-09 16:24:59 +04:00
|
|
|
}
|
|
|
|
|
2002-09-30 07:43:32 +04:00
|
|
|
|
|
|
|
status_t
|
|
|
|
user_delete_sem(sem_id id)
|
2002-07-09 16:24:59 +04:00
|
|
|
{
|
|
|
|
return delete_sem(id);
|
|
|
|
}
|
|
|
|
|
2002-09-30 07:43:32 +04:00
|
|
|
|
|
|
|
status_t
|
2002-10-26 05:11:15 +04:00
|
|
|
user_delete_sem_etc(sem_id id, status_t return_code, bool interrupted)
|
2002-07-09 16:24:59 +04:00
|
|
|
{
|
2002-10-26 05:11:15 +04:00
|
|
|
return delete_sem_etc(id, return_code, interrupted);
|
2002-07-09 16:24:59 +04:00
|
|
|
}
|
|
|
|
|
2002-09-30 07:43:32 +04:00
|
|
|
|
|
|
|
status_t
|
|
|
|
user_acquire_sem(sem_id id)
|
2002-07-09 16:24:59 +04:00
|
|
|
{
|
|
|
|
return user_acquire_sem_etc(id, 1, 0, 0);
|
|
|
|
}
|
|
|
|
|
2002-09-30 07:43:32 +04:00
|
|
|
|
|
|
|
status_t
|
|
|
|
user_acquire_sem_etc(sem_id id, int32 count, uint32 flags, bigtime_t timeout)
|
2002-07-09 16:24:59 +04:00
|
|
|
{
|
2004-02-23 06:54:00 +03:00
|
|
|
return acquire_sem_etc(id, count, flags | B_CAN_INTERRUPT, timeout);
|
2002-07-09 16:24:59 +04:00
|
|
|
}
|
|
|
|
|
2002-09-30 07:43:32 +04:00
|
|
|
|
|
|
|
status_t
|
|
|
|
user_release_sem(sem_id id)
|
2002-07-09 16:24:59 +04:00
|
|
|
{
|
|
|
|
return release_sem_etc(id, 1, 0);
|
|
|
|
}
|
|
|
|
|
2002-09-30 07:43:32 +04:00
|
|
|
|
|
|
|
status_t
|
|
|
|
user_release_sem_etc(sem_id id, int32 count, uint32 flags)
|
2002-07-09 16:24:59 +04:00
|
|
|
{
|
|
|
|
return release_sem_etc(id, count, flags);
|
|
|
|
}
|
|
|
|
|
2002-09-30 07:43:32 +04:00
|
|
|
|
|
|
|
status_t
|
2004-02-23 06:54:00 +03:00
|
|
|
user_get_sem_count(sem_id id, int32 *userCount)
|
2002-07-09 16:24:59 +04:00
|
|
|
{
|
2004-02-23 06:54:00 +03:00
|
|
|
status_t status;
|
|
|
|
int32 count;
|
|
|
|
|
|
|
|
if (userCount == NULL || !IS_USER_ADDRESS(userCount))
|
|
|
|
return B_BAD_ADDRESS;
|
|
|
|
|
|
|
|
status = get_sem_count(id, &count);
|
|
|
|
if (status == B_OK && user_memcpy(userCount, &count, sizeof(int32)) < B_OK)
|
|
|
|
return B_BAD_ADDRESS;
|
|
|
|
|
|
|
|
return status;
|
2002-07-09 16:24:59 +04:00
|
|
|
}
|
|
|
|
|
2002-09-30 07:43:32 +04:00
|
|
|
|
|
|
|
status_t
|
2004-02-23 06:54:00 +03:00
|
|
|
user_get_sem_info(sem_id id, struct sem_info *userInfo, size_t size)
|
2002-07-09 16:24:59 +04:00
|
|
|
{
|
|
|
|
struct sem_info info;
|
2004-02-23 06:54:00 +03:00
|
|
|
status_t status;
|
2002-07-09 16:24:59 +04:00
|
|
|
|
2004-02-23 06:54:00 +03:00
|
|
|
if (userInfo == NULL || !IS_USER_ADDRESS(userInfo))
|
|
|
|
return B_BAD_ADDRESS;
|
2002-07-09 16:24:59 +04:00
|
|
|
|
2004-02-23 06:54:00 +03:00
|
|
|
status = _get_sem_info(id, &info, size);
|
|
|
|
if (status == B_OK && user_memcpy(userInfo, &info, size) < B_OK)
|
|
|
|
return B_BAD_ADDRESS;
|
2002-09-30 07:43:32 +04:00
|
|
|
|
2004-02-23 06:54:00 +03:00
|
|
|
return status;
|
2002-07-09 16:24:59 +04:00
|
|
|
}
|
|
|
|
|
2002-09-30 07:43:32 +04:00
|
|
|
|
|
|
|
status_t
|
2004-02-23 06:54:00 +03:00
|
|
|
user_get_next_sem_info(team_id team, int32 *userCookie, struct sem_info *userInfo,
|
|
|
|
size_t size)
|
2002-07-09 16:24:59 +04:00
|
|
|
{
|
|
|
|
struct sem_info info;
|
2002-09-03 06:09:48 +04:00
|
|
|
int32 cookie;
|
2004-02-23 06:54:00 +03:00
|
|
|
status_t status;
|
|
|
|
|
|
|
|
if (userCookie == NULL || userInfo == NULL
|
|
|
|
|| !IS_USER_ADDRESS(userCookie) || !IS_USER_ADDRESS(userInfo)
|
|
|
|
|| user_memcpy(&cookie, userCookie, sizeof(int32)) < B_OK)
|
|
|
|
return B_BAD_ADDRESS;
|
|
|
|
|
|
|
|
status = _get_next_sem_info(team, &cookie, &info, size);
|
|
|
|
|
|
|
|
if (status == B_OK) {
|
|
|
|
if (user_memcpy(userInfo, &info, size) < B_OK
|
|
|
|
|| user_memcpy(userCookie, &cookie, sizeof(int32)) < B_OK)
|
|
|
|
return B_BAD_ADDRESS;
|
|
|
|
}
|
|
|
|
|
|
|
|
return status;
|
2002-07-09 16:24:59 +04:00
|
|
|
}
|
|
|
|
|
2002-09-30 07:43:32 +04:00
|
|
|
|
|
|
|
status_t
|
2004-02-23 06:54:00 +03:00
|
|
|
user_set_sem_owner(sem_id id, team_id team)
|
2002-07-09 16:24:59 +04:00
|
|
|
{
|
2004-02-23 06:54:00 +03:00
|
|
|
return set_sem_owner(id, team);
|
2002-07-09 16:24:59 +04:00
|
|
|
}
|