Hi,
I'm looking for solution, how to make atomic execution block on Cortex-M3 uC. What is the best solution for this?
Best regards, Lukas
With these pipelined cores it often is possible that an interrupt is still taken between the instruction disabling the interrupts and the next instruction.
The exact workings of the interrupt system of a MCU should be described, in great detail, in the datasheet.
Also, if the interrupt is serviced between the "disable interrupts" instruction and the next (which is the first of the atomic block), then the atomic block is still that - atomic. It's not being interrupted.
The problem is that some processors will perform at least one instruction after the "disable interrupt" instruction, so one or more nop instructions may be needed.
The exact workings of the interrupt system is seldom fully documented in the data sheets :(
Sometimes the information about the need for nop instructions are only available in application notes. And sometimes the application notes are not even for the specific processor, but for a different processor in the same - or similar - family. In the end, you may have to call the manufacturers support engineers to get the answer...
On some processors, the disable interrupt function might take sometime to take effect. This is because the masking is done via control register accesses through the bus. (there could be wait state on the bus and the write operation could be buffered, and the interrupt controller might need a cycle to update to the new setting).
On Cortex-M3, the interrupt mask registers are inside the processors (e.g. FAULTMASK, PRIMASK). As soon as PRIMASK/FAULTMASK/BASEPRI is set, the effect come immediately. So if you set the PRIMASK and the interrupt arrived at the same time, the interrupt will have to wait until the mask is cleared. To do this, the interrupt mask registers are accessible by CPS, MSR and MRS instructions only, for example:
CPSID i ; Set PRIMASK, disable interrupts CPSIE i ; Clear PRIMASK, enable interrupts CPSID f ; Set FAULTMASK, disable interrupts & hard fault handler CPSIE f ; Clear FAULTMASK, enable interrupts & ; hard fault handler
In C, the function
__disable_irq() translate to CPSID i
and
__enable_irq() translate to CPSIE i
These functions are also available for other ARM cores (e.g. ARM7TDMI). But instead of changing PRIMASK, __disable_irq change the "I" bit in CPSR.