Fix some bad SMP logic in sched_removereadytorun.c

This commit is contained in:
Gregory Nutt 2016-02-20 08:07:54 -06:00
parent 883a1adfe2
commit c929db42f9
2 changed files with 129 additions and 38 deletions

View File

@ -369,7 +369,7 @@ bool sched_addreadytorun(FAR struct tcb_s *btcb)
btcb->task_state = TSTATE_TASK_ASSIGNED; btcb->task_state = TSTATE_TASK_ASSIGNED;
} }
/* All done, restart the other that CPU. */ /* All done, restart the other CPU (if it was paused). */
if (cpu != me) if (cpu != me)
{ {

View File

@ -72,15 +72,84 @@
* *
****************************************************************************/ ****************************************************************************/
#ifndef CONFIG_SMP
bool sched_removereadytorun(FAR struct tcb_s *rtcb)
{
bool doswitch = false;
/* Check if the TCB to be removed is at the head of the ready to run list.
* There is only one list, g_readytorun, and it always contains the
* currently running task. If we are removing the head of this list,
* then we are removing the currently active task.
*/
if (rtcb->blink == NULL)
{
/* There must always be at least one task in the list (the IDLE task)
* after the TCB being removed.
*/
FAR struct tcb_s *ntcb = (FAR struct tcb_s *)rtcb->flink;
DEBUGASSERT(ntcb != NULL);
/* Inform the instrumentation layer that we are switching tasks */
sched_note_switch(rtcb, ntcb);
ntcb->task_state = TSTATE_TASK_RUNNING;
doswitch = true;
}
/* Remove the TCB from the ready-to-run list. In the non-SMP case, this
* is always the g_readytorun list.
*/
dq_rem((FAR dq_entry_t *)rtcb, (FAR dq_queue_t *)&g_readytorun);
/* Since the TCB is not in any list, it is now invalid */
rtcb->task_state = TSTATE_TASK_INVALID;
return doswitch;
}
#endif /* !CONFIG_SMP */
/****************************************************************************
* Name: sched_removereadytorun
*
* Description:
* This function removes a TCB from the ready to run list.
*
* Inputs:
* rtcb - Points to the TCB that is ready-to-run
*
* Return Value:
* true if the currently active task (the head of the ready-to-run list)
* has changed.
*
* Assumptions:
* - The caller has established a critical section before calling this
* function (calling sched_lock() first is NOT a good idea -- use
* enter_critical_section()).
* - The caller handles the condition that occurs if the head of the
* ready-to-run list is changed.
*
****************************************************************************/
#ifdef CONFIG_SMP
bool sched_removereadytorun(FAR struct tcb_s *rtcb) bool sched_removereadytorun(FAR struct tcb_s *rtcb)
{ {
FAR dq_queue_t *tasklist; FAR dq_queue_t *tasklist;
bool ret = false; bool doswitch = false;
int cpu;
/* Check if the TCB to be removed is at the head of a ready to run list. */ /* Which CPU (if any) is the task running on? Which task list holds the
* TCB?
*/
#ifdef CONFIG_SMP cpu = rtcb->cpu;
/* For the case of SMP, there are two lists involved: (1) the tasklist = TLIST_HEAD(rtcb->task_state, cpu);
/* Check if the TCB to be removed is at the head of a ready to run list.
* For the case of SMP, there are two lists involved: (1) the
* g_readytorun list that holds non-running tasks that have not been * g_readytorun list that holds non-running tasks that have not been
* assigned to a CPU, and (2) and the g_assignedtasks[] lists which hold * assigned to a CPU, and (2) and the g_assignedtasks[] lists which hold
* tasks assigned a CPU, including the task that is currently running on * tasks assigned a CPU, including the task that is currently running on
@ -94,21 +163,27 @@ bool sched_removereadytorun(FAR struct tcb_s *rtcb)
*/ */
if (rtcb->blink == NULL && TLIST_ISRUNNABLE(rtcb->task_state)) if (rtcb->blink == NULL && TLIST_ISRUNNABLE(rtcb->task_state))
#else {
/* There is only one list, g_readytorun, and it always contains the FAR struct tcb_s *ntcb;
* currently running task. If we are removing the head of this list, int me;
* then we are removing the currently active task.
/* There must always be at least one task in the list (the IDLE task)
* after the TCB being removed.
*/ */
if (rtcb->blink == NULL) ntcb = (FAR struct tcb_s *)rtcb->flink;
#endif
{
/* There must always be at least one task in the list (the idle task) */
FAR struct tcb_s *ntcb = (FAR struct tcb_s *)rtcb->flink;
DEBUGASSERT(ntcb != NULL); DEBUGASSERT(ntcb != NULL);
#ifdef CONFIG_SMP /* If we are modifying the head of some assigned task list other than
* our own, we will need to stop that CPU.
*/
me = this_cpu();
if (cpu != me)
{
DEBUGVERIFY(up_cpu_pause(cpu));
}
/* Will pre-emption be disabled after the switch? If the lockcount is /* Will pre-emption be disabled after the switch? If the lockcount is
* greater than zero, then this task/this CPU holds the scheduler lock. * greater than zero, then this task/this CPU holds the scheduler lock.
*/ */
@ -117,18 +192,18 @@ bool sched_removereadytorun(FAR struct tcb_s *rtcb)
{ {
/* Yes... make sure that scheduling logic knows about this */ /* Yes... make sure that scheduling logic knows about this */
spin_setbit(&g_cpu_lockset, this_cpu(), &g_cpu_locksetlock, spin_setbit(&g_cpu_lockset, cpu, &g_cpu_locksetlock,
&g_cpu_schedlock); &g_cpu_schedlock);
} }
else else
{ {
/* No.. we may need to perform release our hold on the lock. */ /* No.. we may need to perform release our hold on the lock. */
spin_clrbit(&g_cpu_lockset, this_cpu(), &g_cpu_locksetlock, spin_clrbit(&g_cpu_lockset, cpu, &g_cpu_locksetlock,
&g_cpu_schedlock); &g_cpu_schedlock);
} }
/* Interrupts be disabled after the switch. If irqcount is greater /* Interrupts may be disabled after the switch. If irqcount is greater
* than zero, then this task/this CPU holds the IRQ lock * than zero, then this task/this CPU holds the IRQ lock
*/ */
@ -136,40 +211,56 @@ bool sched_removereadytorun(FAR struct tcb_s *rtcb)
{ {
/* Yes... make sure that scheduling logic knows about this */ /* Yes... make sure that scheduling logic knows about this */
spin_setbit(&g_cpu_irqset, this_cpu(), &g_cpu_irqsetlock, spin_setbit(&g_cpu_irqset, cpu, &g_cpu_irqsetlock,
&g_cpu_irqlock); &g_cpu_irqlock);
} }
else else
{ {
/* No.. we may need to perform release our hold on the lock. */ /* No.. we may need to perform release our hold on the lock. */
spin_setbit(&g_cpu_irqset, this_cpu(), &g_cpu_irqsetlock, spin_setbit(&g_cpu_irqset, cpu, &g_cpu_irqsetlock,
&g_cpu_irqlock); &g_cpu_irqlock);
} }
#endif
/* Inform the instrumentation layer that we are switching tasks */ /* Inform the instrumentation layer that we are switching tasks */
sched_note_switch(rtcb, ntcb); sched_note_switch(rtcb, ntcb);
ntcb->task_state = TSTATE_TASK_RUNNING; ntcb->task_state = TSTATE_TASK_RUNNING;
ret = true;
}
/* Remove the TCB from the ready-to-run list. In the non-SMP case, this /* The task is running but the CPU that it was running on has been
* is always the g_readytorun list; In the SMP case, however, this may be * paused. We can now safely remove its TCB from the ready-to-run
* either the g_readytorun() or the g_assignedtasks[cpu] list. * task list. In the SMP case this may be either the g_readytorun()
* or the g_assignedtasks[cpu] list.
*/ */
#ifdef CONFIG_SMP
tasklist = TLIST_HEAD(rtcb->task_state, rtcb->cpu);
#else
tasklist = (FAR dq_queue_t *)&g_readytorun;
#endif
dq_rem((FAR dq_entry_t *)rtcb, tasklist); dq_rem((FAR dq_entry_t *)rtcb, tasklist);
/* Since the TCB is not in any list, it is now invalid */ /* All done, restart the other CPU (if it was paused). */
doswitch = true;
if (cpu != me)
{
/* In this we will not want to report a context switch to this
* CPU. Only the other CPU is affected.
*/
DEBUGVERIFY(up_cpu_resume(cpu));
doswitch = false;
}
}
else
{
/* The task is not running. Just remove its TCB from the ready-to-run
* list. In the SMP case this may be either the g_readytorun() or the
* g_assignedtasks[cpu] list.
*/
dq_rem((FAR dq_entry_t *)rtcb, tasklist);
}
/* Since the TCB is no longer in any list, it is now invalid */
rtcb->task_state = TSTATE_TASK_INVALID; rtcb->task_state = TSTATE_TASK_INVALID;
return ret; return doswitch;
} }
#endif /* CONFIG_SMP */