sched: implement waitqueues
This commit is contained in:
@@ -12,10 +12,27 @@
|
|||||||
#define PRIO_MAX 32
|
#define PRIO_MAX 32
|
||||||
#define THREAD_KSTACK_ORDER VM_PAGE_4K
|
#define THREAD_KSTACK_ORDER VM_PAGE_4K
|
||||||
|
|
||||||
|
#define wait_event(wq, cond) \
|
||||||
|
({ \
|
||||||
|
struct thread *self = current_thread(); \
|
||||||
|
struct wait_item waiter; \
|
||||||
|
wait_item_init(&waiter, self); \
|
||||||
|
for (;;) { \
|
||||||
|
thread_wait_begin(&waiter, wq); \
|
||||||
|
if (cond) { \
|
||||||
|
break; \
|
||||||
|
} \
|
||||||
|
schedule(SCHED_NORMAL); \
|
||||||
|
} \
|
||||||
|
thread_wait_end(&waiter, wq); \
|
||||||
|
})
|
||||||
|
|
||||||
#ifdef __cplusplus
|
#ifdef __cplusplus
|
||||||
extern "C" {
|
extern "C" {
|
||||||
#endif
|
#endif
|
||||||
|
|
||||||
|
struct runqueue;
|
||||||
|
|
||||||
enum task_state {
|
enum task_state {
|
||||||
TASK_RUNNING,
|
TASK_RUNNING,
|
||||||
TASK_STOPPED,
|
TASK_STOPPED,
|
||||||
@@ -83,6 +100,8 @@ struct thread {
|
|||||||
|
|
||||||
uintptr_t tr_sp, tr_bp;
|
uintptr_t tr_sp, tr_bp;
|
||||||
|
|
||||||
|
struct runqueue *tr_rq;
|
||||||
|
|
||||||
struct queue_entry tr_threads;
|
struct queue_entry tr_threads;
|
||||||
struct queue_entry tr_rqentry;
|
struct queue_entry tr_rqentry;
|
||||||
|
|
||||||
@@ -174,6 +193,13 @@ extern void remove_timer(struct timer *timer);
|
|||||||
extern unsigned long schedule_timeout(unsigned long clock_ticks);
|
extern unsigned long schedule_timeout(unsigned long clock_ticks);
|
||||||
extern unsigned long milli_sleep(unsigned long ms);
|
extern unsigned long milli_sleep(unsigned long ms);
|
||||||
|
|
||||||
|
extern void wait_item_init(struct wait_item *item, struct thread *thr);
|
||||||
|
extern void thread_wait_begin(struct wait_item *waiter, struct waitqueue *q);
|
||||||
|
extern void thread_wait_end(struct wait_item *waiter, struct waitqueue *q);
|
||||||
|
extern void wait_on_queue(struct waitqueue *q);
|
||||||
|
extern void wakeup_queue(struct waitqueue *q);
|
||||||
|
extern void wakeup_one(struct waitqueue *q);
|
||||||
|
|
||||||
#ifdef __cplusplus
|
#ifdef __cplusplus
|
||||||
}
|
}
|
||||||
#endif
|
#endif
|
||||||
|
|||||||
@@ -50,6 +50,7 @@ void rq_enqueue(struct runqueue *rq, struct thread *thr)
|
|||||||
rq->rq_nthreads++;
|
rq->rq_nthreads++;
|
||||||
|
|
||||||
rq->rq_readybits |= PRIO_MASK(thread_priority(thr));
|
rq->rq_readybits |= PRIO_MASK(thread_priority(thr));
|
||||||
|
thr->tr_rq = rq;
|
||||||
}
|
}
|
||||||
|
|
||||||
struct runqueue *cpu_rq(unsigned int cpu)
|
struct runqueue *cpu_rq(unsigned int cpu)
|
||||||
|
|||||||
73
sched/wait.c
Normal file
73
sched/wait.c
Normal file
@@ -0,0 +1,73 @@
|
|||||||
|
#include <socks/sched.h>
|
||||||
|
#include <socks/cpu.h>
|
||||||
|
|
||||||
|
void wait_item_init(struct wait_item *item, struct thread *thr)
|
||||||
|
{
|
||||||
|
item->w_thread = thr;
|
||||||
|
item->w_entry = QUEUE_ENTRY_INIT;
|
||||||
|
}
|
||||||
|
|
||||||
|
void thread_wait_begin(struct wait_item *waiter, struct waitqueue *q)
|
||||||
|
{
|
||||||
|
unsigned long flags;
|
||||||
|
spin_lock_irqsave(&q->wq_lock, &flags);
|
||||||
|
|
||||||
|
queue_push_back(&q->wq_waiters, &waiter->w_entry);
|
||||||
|
waiter->w_thread->tr_state = THREAD_SLEEPING;
|
||||||
|
|
||||||
|
spin_unlock_irqrestore(&q->wq_lock, flags);
|
||||||
|
}
|
||||||
|
|
||||||
|
void thread_wait_end(struct wait_item *waiter, struct waitqueue *q)
|
||||||
|
{
|
||||||
|
waiter->w_thread->tr_state = THREAD_READY;
|
||||||
|
|
||||||
|
unsigned long flags;
|
||||||
|
spin_lock_irqsave(&q->wq_lock, &flags);
|
||||||
|
|
||||||
|
queue_delete(&q->wq_waiters, &waiter->w_entry);
|
||||||
|
|
||||||
|
spin_unlock_irqrestore(&q->wq_lock, flags);
|
||||||
|
}
|
||||||
|
|
||||||
|
void wakeup_queue(struct waitqueue *q)
|
||||||
|
{
|
||||||
|
unsigned long flags;
|
||||||
|
spin_lock_irqsave(&q->wq_lock, &flags);
|
||||||
|
struct queue_entry *ent = queue_pop_front(&q->wq_waiters);
|
||||||
|
while (ent) {
|
||||||
|
struct wait_item *waiter = QUEUE_CONTAINER(struct wait_item, w_entry, ent);
|
||||||
|
struct thread *thr = waiter->w_thread;
|
||||||
|
struct runqueue *rq = thr->tr_rq;
|
||||||
|
if (!rq) {
|
||||||
|
rq = cpu_rq(this_cpu());
|
||||||
|
}
|
||||||
|
|
||||||
|
thr->tr_state = THREAD_READY;
|
||||||
|
rq_enqueue(rq, thr);
|
||||||
|
|
||||||
|
ent = queue_pop_front(&q->wq_waiters);
|
||||||
|
}
|
||||||
|
|
||||||
|
spin_unlock_irqrestore(&q->wq_lock, flags);
|
||||||
|
}
|
||||||
|
|
||||||
|
void wakeup_one(struct waitqueue *q)
|
||||||
|
{
|
||||||
|
unsigned long flags;
|
||||||
|
spin_lock_irqsave(&q->wq_lock, &flags);
|
||||||
|
struct queue_entry *ent = queue_pop_front(&q->wq_waiters);
|
||||||
|
if(ent) {
|
||||||
|
struct wait_item *waiter = QUEUE_CONTAINER(struct wait_item, w_entry, ent);
|
||||||
|
struct thread *thr = waiter->w_thread;
|
||||||
|
struct runqueue *rq = thr->tr_rq;
|
||||||
|
if (!rq) {
|
||||||
|
rq = cpu_rq(this_cpu());
|
||||||
|
}
|
||||||
|
|
||||||
|
thr->tr_state = THREAD_READY;
|
||||||
|
rq_enqueue(rq, thr);
|
||||||
|
}
|
||||||
|
|
||||||
|
spin_unlock_irqrestore(&q->wq_lock, flags);
|
||||||
|
}
|
||||||
Reference in New Issue
Block a user