do_migrate(): signal migrated thread before releasing runq lock

This commit is contained in:
Balazs Gerofi
2017-01-25 14:39:26 +09:00
parent b812848a0e
commit addbe91e59
4 changed files with 51 additions and 15 deletions

View File

@@ -790,6 +790,7 @@ void spin_sleep_or_schedule(void);
void runq_add_thread(struct thread *thread, int cpu_id); void runq_add_thread(struct thread *thread, int cpu_id);
void runq_del_thread(struct thread *thread, int cpu_id); void runq_del_thread(struct thread *thread, int cpu_id);
int sched_wakeup_thread(struct thread *thread, int valid_states); int sched_wakeup_thread(struct thread *thread, int valid_states);
int sched_wakeup_thread_locked(struct thread *thread, int valid_states);
void sched_request_migrate(int cpu_id, struct thread *thread); void sched_request_migrate(int cpu_id, struct thread *thread);
void check_need_resched(void); void check_need_resched(void);

View File

@@ -27,6 +27,8 @@ typedef int (*waitq_func_t)(struct waitq_entry *wait, unsigned mode,
int default_wake_function(struct waitq_entry *wait, unsigned mode, int flags, int default_wake_function(struct waitq_entry *wait, unsigned mode, int flags,
void *key); void *key);
int locked_wake_function(struct waitq_entry *wait, unsigned mode, int flags,
void *key);
typedef struct waitq { typedef struct waitq {
ihk_spinlock_t lock; ihk_spinlock_t lock;
@@ -57,6 +59,13 @@ typedef struct waitq_entry {
.link = { &(name).link, &(name).link } \ .link = { &(name).link, &(name).link } \
} }
#define DECLARE_WAITQ_ENTRY_LOCKED(name, tsk) \
waitq_entry_t name = { \
.private = tsk, \
.func = locked_wake_function, \
.link = { &(name).link, &(name).link } \
}
extern void waitq_init(waitq_t *waitq); extern void waitq_init(waitq_t *waitq);
extern void waitq_init_entry(waitq_entry_t *entry, struct thread *proc); extern void waitq_init_entry(waitq_entry_t *entry, struct thread *proc);
extern int waitq_active(waitq_t *waitq); extern int waitq_active(waitq_t *waitq);

View File

@@ -2658,9 +2658,9 @@ static void do_migrate(void)
__FUNCTION__, req->thread->tid, old_cpu_id, cpu_id); __FUNCTION__, req->thread->tid, old_cpu_id, cpu_id);
v->flags |= CPU_FLAG_NEED_RESCHED; v->flags |= CPU_FLAG_NEED_RESCHED;
ihk_mc_interrupt_cpu(get_x86_cpu_local_variable(cpu_id)->apic_id, 0xd1); waitq_wakeup(&req->wq);
double_rq_unlock(cur_v, v, irqstate); double_rq_unlock(cur_v, v, irqstate);
continue;
ack: ack:
waitq_wakeup(&req->wq); waitq_wakeup(&req->wq);
} }
@@ -2908,8 +2908,8 @@ void check_need_resched(void)
} }
} }
int int __sched_wakeup_thread(struct thread *thread,
sched_wakeup_thread(struct thread *thread, int valid_states) int valid_states, int runq_locked)
{ {
int status; int status;
unsigned long irqstate; unsigned long irqstate;
@@ -2917,20 +2917,25 @@ sched_wakeup_thread(struct thread *thread, int valid_states)
struct process *proc = thread->proc; struct process *proc = thread->proc;
struct mcs_rwlock_node updatelock; struct mcs_rwlock_node updatelock;
dkprintf("sched_wakeup_process,proc->pid=%d,valid_states=%08x,proc->status=%08x,proc->cpu_id=%d,my cpu_id=%d\n", dkprintf("%s: proc->pid=%d, valid_states=%08x, "
proc->pid, valid_states, thread->status, thread->cpu_id, ihk_mc_get_processor_id()); "proc->status=%08x, proc->cpu_id=%d,my cpu_id=%d\n",
__FUNCTION__,
proc->pid, valid_states, thread->status,
thread->cpu_id, ihk_mc_get_processor_id());
irqstate = ihk_mc_spinlock_lock(&(thread->spin_sleep_lock)); irqstate = ihk_mc_spinlock_lock(&(thread->spin_sleep_lock));
if (thread->spin_sleep == 1) { if (thread->spin_sleep == 1) {
dkprintf("sched_wakeup_process() spin wakeup: cpu_id: %d\n", dkprintf("%s: spin wakeup: cpu_id: %d\n",
thread->cpu_id); __FUNCTION__, thread->cpu_id);
status = 0; status = 0;
} }
thread->spin_sleep = 0; thread->spin_sleep = 0;
ihk_mc_spinlock_unlock(&(thread->spin_sleep_lock), irqstate); ihk_mc_spinlock_unlock(&(thread->spin_sleep_lock), irqstate);
if (!runq_locked) {
irqstate = ihk_mc_spinlock_lock(&(v->runq_lock)); irqstate = ihk_mc_spinlock_lock(&(v->runq_lock));
}
if (thread->status & valid_states) { if (thread->status & valid_states) {
mcs_rwlock_writer_lock_noirq(&proc->update_lock, &updatelock); mcs_rwlock_writer_lock_noirq(&proc->update_lock, &updatelock);
@@ -2944,18 +2949,32 @@ sched_wakeup_thread(struct thread *thread, int valid_states)
status = -EINVAL; status = -EINVAL;
} }
if (!runq_locked) {
ihk_mc_spinlock_unlock(&(v->runq_lock), irqstate); ihk_mc_spinlock_unlock(&(v->runq_lock), irqstate);
}
if (!status && (thread->cpu_id != ihk_mc_get_processor_id())) { if (!status && (thread->cpu_id != ihk_mc_get_processor_id())) {
dkprintf("sched_wakeup_process,issuing IPI,thread->cpu_id=%d\n", dkprintf("%s: issuing IPI, thread->cpu_id=%d\n",
thread->cpu_id); __FUNCTION__, thread->cpu_id);
ihk_mc_interrupt_cpu(get_x86_cpu_local_variable(thread->cpu_id)->apic_id, ihk_mc_interrupt_cpu(
get_x86_cpu_local_variable(thread->cpu_id)->apic_id,
0xd1); 0xd1);
} }
return status; return status;
} }
int sched_wakeup_thread_locked(struct thread *thread, int valid_states)
{
return __sched_wakeup_thread(thread, valid_states, 1);
}
int sched_wakeup_thread(struct thread *thread, int valid_states)
{
return __sched_wakeup_thread(thread, valid_states, 0);
}
/* /*
* 1. Add current process to waitq * 1. Add current process to waitq
* 2. Queue migration request into the target CPU's queue * 2. Queue migration request into the target CPU's queue
@@ -2979,7 +2998,7 @@ void sched_request_migrate(int cpu_id, struct thread *thread)
struct cpu_local_var *v = get_cpu_local_var(cpu_id); struct cpu_local_var *v = get_cpu_local_var(cpu_id);
struct migrate_request req = { .thread = thread }; struct migrate_request req = { .thread = thread };
unsigned long irqstate; unsigned long irqstate;
DECLARE_WAITQ_ENTRY(entry, cpu_local_var(current)); DECLARE_WAITQ_ENTRY_LOCKED(entry, cpu_local_var(current));
waitq_init(&req.wq); waitq_init(&req.wq);
waitq_prepare_to_wait(&req.wq, &entry, PS_UNINTERRUPTIBLE); waitq_prepare_to_wait(&req.wq, &entry, PS_UNINTERRUPTIBLE);

View File

@@ -22,6 +22,13 @@ default_wake_function(waitq_entry_t *entry, unsigned mode,
return sched_wakeup_thread(entry->private, PS_NORMAL); return sched_wakeup_thread(entry->private, PS_NORMAL);
} }
int
locked_wake_function(waitq_entry_t *entry, unsigned mode,
int flags, void *key)
{
return sched_wakeup_thread_locked(entry->private, PS_NORMAL);
}
void void
waitq_init(waitq_t *waitq) waitq_init(waitq_t *waitq)
{ {