How to Determine If Cortex-M Processor Is Executing Interrupt Function?

Follow+Star Public Account, don’t miss wonderful content

How to Determine If Cortex-M Processor Is Executing Interrupt Function?

Author | strongerHuang

WeChat Official Account | Embedded Column

Today, I want to share some knowledge I encountered while debugging code: the usage of __get_CONTROL, and the differences between xQueueSend and xQueueSendFromISR;

1

Source of the Problem

I previously ported some code written by others on the FreeRTOS system. After carefully reviewing the source code before the porting, confirming there were no issues, I compiled, downloaded, and ran it, only to suddenly find it “frozen”······

So, I confirmed the ported code again and found no bugs. At this point, I enabled online debugging and found that the program was stuck in the assertion statement in the vPortEnterCritical function. As follows:

How to Determine If Cortex-M Processor Is Executing Interrupt Function?

2

Process of Solving the Problem

I solved the problem by following a conventional thought process, tracking step by step. Many issues are actually similar in principle and can be traced.

1. Check what configASSERT does?

Tracking the code:

#define configASSERT( x ) if( ( x ) == 0 ) { taskDISABLE_INTERRUPTS(); for( ;; ); }
Here, taskDISABLE_INTERRUPTS(); means to disable interrupts. Right after that, for( ;; ); is executed.
At this point, I understood that the program was stuck in the for( ;; ); loop.

2. Further Investigate the Problem

I began to think, why did it execute to this point? Why did it execute:
portDISABLE_INTERRUPTS(); uxCriticalNesting++;   if( uxCriticalNesting == 1 )
these statements?
This is what we commonly refer to as a “critical section.” I understood this when I studied RTOS, and this function must have been called. Therefore, I focused on the parameter portNVIC_INT_CTRL_REG:
#define portNVIC_INT_CTRL_REG     ( * ( ( volatile uint32_t * ) 0xe000ed04 ) )

0xe000ed04? This address, as those familiar with NVIC would know, is the Interrupt Control State Register.

3. Identify the Problem Point

From the above analysis, the problem has already emerged. I then checked relevant content in the Cortex-M3 authoritative guide. (PS: This manual can really solve many problems, translating it into Chinese is a good thing for most friends)
Actually, there is a register: Control Register (CONTROL), which explains this very clearly:

How to Determine If Cortex-M Processor Is Executing Interrupt Function?

As shown in the figure, the general idea is: in interrupt mode, CONTROL[1] is 0. So, I turned my attention to the core_cm3.c file’s source code:
__ASM uint32_t __get_CONTROL(void)
{
  mrs r0, control
  bx lr
}
Those who understand a bit of assembly should already understand that this roughly means accessing the control register state, which is what I mentioned at the beginning, to let everyone understand __get_CONTROL.

4. Online Debugging, Analysis Conclusion

From the analysis above regarding the control register CONTROL, we need to verify whether it meets our expected effect. Through online debugging, we can conclude, as shown in the following two images:

a. Value 0x02 in non-interrupt condition

How to Determine If Cortex-M Processor Is Executing Interrupt Function?

b. Value 0x00 in interrupt condition

How to Determine If Cortex-M Processor Is Executing Interrupt Function?

At this point, the problem has been identified as CONTROL.

3

Application of get_CONTROL

In general, in RTOS real-time operating systems, queues are often used to handle our data, which is commonly referred to as FIFO (First In, First Out).

For example: in the FreeRTOS system, to add data from UART to the queue, we add to the queue in interrupts and in non-interrupts. At this point, we need to use get_CONTROL to determine whether we are currently in an interrupt function.

Of course, there are many similar situations, such as CAN, I2C, SPI, and the same principle applies.

For example, sending data to the queue via CAN bus:

How to Determine If Cortex-M Processor Is Executing Interrupt Function?

This is the content for today, I hope it helps you.
———— END ————

Reply ‘Cortex-M’ or ‘Microcontroller’ in the background to read more related articles.

Welcome to follow my public account, reply “Join Group” to join the technical exchange group according to the rules, reply “1024” to see more content.
Welcome to follow my video account:

How to Determine If Cortex-M Processor Is Executing Interrupt Function?

Click “Read Original” to see more shares, and feel free to share, save, like, and view.

Leave a Comment