blob: fa12c73b7b1f261aebd157c3315e822e6ebf4df3 [file] [log] [blame]
===========================
Notes on AVR context switch
===========================
This document describes the ways and circumstances in which context
switches happen in AVR MCUs.
Used terms and context switch basics
====================================
Context creation
----------------
There are two ways context is created when a task is suspended.
Either the task is suspended in response to a hardware interrupt
(context created ``in-interrupt`` in the following text), or it is
suspended voluntarily, eg. by calling sleep(), read() etc.
(``in-task`` context.)
The resulting context is identical and interchangeable with two differences:
- ``SREG`` - ``in-interrupt`` context has global interrupt enabled ("I"-flag) set
- position in the program where the task resumes running (arbitrary point for ``in-interrupt`` vs. inside ``up_switch_context()`` for ``in-task``
Task resumption
---------------
Task can be resumed in two corresponding situations - context switch
in response to a hardware interrupt (``by-interrupt``) or in response
to other task relinquishing the CPU (``by-task``)
Context switch
==============
Two ways of context creation combined with two ways of task resumption
give 4 possibilities of context switch process, two of which are
not interesting:
1st combination
---------------
``in-task`` context resumed in ``by-task`` context switch. Context
to be resumed has "I" flag cleared and SREG is restored with that flag cleared.
The ``ret`` instruction is used to resume the task,"returning" to the point
where it gave the CPU up, which is inside ``up_switch_context()``.
This function is supposed to be executed with interrupt disabled ("This
function is called only from the NuttX scheduling logic. Interrupts
will always be disabled when this function is
called." https://nuttx.apache.org/docs/latest/reference/os/arch.html ) Caller
of the context switch method is therefore responsible
for re-enabling interrupts.
2nd
---
``in-interrupt`` resumed in ``by-interrupt`` context switch. The task
essentially (from its point of view) exits interrupt handler after
it entered it. Instruction ``reti`` is used to return from the handler,
setting "I"-flag in the process.
3rd
---
Third and fourth combinations are more interesting:
``in-task`` context resumed in ``by-interrupt``. The CPU enters
ISR and regular program flow requires returning from ISR and setting
"I"-flag by ``reti`` which does not happen. Task is resumed with
interrupts disabled. However, it is resumed inside ``up_switch_context()``
and caller of that function will set the "I"-flag at some point.
Task then runs with interrupts enabled, all is well.
4th
---
``in-interrupt`` resumed in ``by-task``, ie. in ``up_switch_context()``.
``reti`` is used to resume the task, setting "I"-flag in the process.
This would be incorrect for ``up_switch_context()`` - it is supposed
to run with interrupts disabled - but the task resumes running
from the point where it was interrupted, which is not inside
of ``up_switch_context()``. All is well.
AVRDx core considerations
=========================
Now, all of the above holds true for eg. ATmega chips which control
interrupt execution solely by the "I"-flag, allowing the code
to not care about where the context switch was triggered. Regardless
of that, the MCU will always end up in correct state even if the
context switch cause doesn't match the context being restored
(cases 3 and 4.)
This is not the case for AVR Dx family which behaves differently.
The interrupt controller does respect the "I"-flag in a sense where
it considers interrupts disabled when the flag is cleared. However,
it is possible that interrupts are not enabled when the flag is set.
That depends on a logical AND between "I"-flag and "interrupt handler
is not executing" internal state. (Refer to the documentation
for more precise explanation.)
What this means is that if eg. ``in-task`` context gets
resumed in ``by-interrupt`` condition (case 3 above), then ``ret``
instruction is used
to resume the task. As discussed above, the "I"-flag is not set this
way but that is not a problem, it is eventually set later. However,
the internal state "running the interrupt handler" is not cleared.
This means that the task keeps running with "global interrupts are
enabled" but is actually unable to be interrupted. The context switch
code needs to handle this.
Conversely, there is a similar problem with ``in-interrupt`` context
being resumed in ``by-task`` (case 4). Instruction ``reti`` is used
but there is no internal state to be cleared. Unlike the previous case,
no problem related to this was observed but it still looks like something
the code wants to avoid. It could trigger all sorts of undefined behaviour
in the chip otherwise.