This discussion has been locked.
You can no longer post new replies to this discussion. If you have a question you can start a new discussion

Atomic block - intterupts proof on cortex-M3

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

Parents Reply Children
  • I am always suspicious about disabling/enabling interrupts to make a piece of code atomic. With these pipelined cores it often is possible that an interrupt is still taken between the instruction disabling the interrupts and the next instruction. If/when this happens, the whole interrupt service is run with interrupts disabled, and then the atomic code. No problem if you don't have any time critical interrupts, but if you do, this may cause you to miss the deadline. I don't know if this applies for the Cortex-M3 though.

  • 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.