stm32中优先使用原子操作的具体实现方式
在STM32中,优先使用的原子操作主要包括位带操作、LDREX/STREX指令以及CMSIS提供的原子操作宏,以下是具体实现方式12:
1. 位带操作(Bit-Banding)
通过硬件直接对单个比特进行原子读写,适用于GPIO或寄存器位操作1:
#define BITBAND(addr, bitnum) ((addr & 0xF0000000) + 0x2000000 + ((addr & 0xFFFFF) << 5) + (bitnum << 2))
#define MEM_ADDR(addr) *((volatile uint32_t *)(addr))// 原子设置GPIOB的PIN5为高电平
MEM_ADDR(BITBAND(&GPIOB->ODR, 5)) = 1;
2. LDREX/STREX指令
基于Cortex-M的独占访问指令,适用于共享变量修改23:
volatile uint32_t shared_var = 0;void atomic_increment(void) {uint32_t val;do {val = __LDREXW(&shared_var); // 独占加载val += 1;} while (__STREXW(val, &shared_var)); // 独占存储,失败则重试__CLREX(); // 清除独占标记
}
3. CMSIS原子操作宏
CMSIS库提供的封装宏,如ATOMIC_SET_BIT
和ATOMIC_MODIFY_REG
24:
#include "core_cm4.h"// 原子设置USART1的CR1寄存器第3位
ATOMIC_SET_BIT(USART1->CR1, USART_CR1_TXEIE);// 原子修改TIM2的CCER寄存器
ATOMIC_MODIFY_REG(TIM2->CCER, TIM_CCER_CC1E, 0x1);
#include "core_cm4.h"// 原子设置USART1的CR1寄存器位
ATOMIC_SET_BIT(USART1->CR1, USART_CR1_TXEIE);// 原子修改变量
uint32_t val = 0;
ATOMIC_MODIFY_REG(val, 0xFF, 0x55); // 将低8位改为0x55
4. GPIO的BSRR寄存器
通过置位/复位寄存器实现GPIO的原子操作1:
// 原子设置PB5高电平、PB6低电平
GPIOB->BSRR = (1 << 5) | (1 << (6 + 16));
临界区保护
HAL库提供__disable_irq()
和__enable_irq()
函数,可临时禁用中断实现原子性
volatile uint32_t counter = 0;void atomic_increment(void) {__disable_irq(); // 进入临界区counter++; // 原子操作__enable_irq(); // 退出临界区
}
代码实现示例
以下是一个综合示例,展示原子操作在多线程环境中的应用35:
atomic_ops.c
已完成
此代码演示了计数器原子递增和GPIO标志位原子设置,适用于中断与主程序共享数据的场景1
#include "stm32f4xx.h"
#include "core_cm4.h"volatile uint32_t counter = 0;// 使用LDREX/STREX原子递增
void atomic_counter_inc(void) {uint32_t val;do {val = __LDREXW(&counter);val++;} while (__STREXW(val, &counter));__CLREX();
}// 使用CMSIS宏原子设置标志位
void atomic_flag_set(volatile uint32_t *reg, uint32_t bit) {ATOMIC_SET_BIT(*reg, bit);
}int main(void) {atomic_counter_inc();atomic_flag_set(&GPIOA->ODR, 5);return 0;
}
#include "stm32f4xx.h"
#include "core_cm4.h"volatile uint32_t global_var = 0; // 待修改的全局变量void atomic_write(uint32_t new_value) {uint32_t current_val;do {current_val = __LDREXW(&global_var); // 独占加载当前值} while (__STREXW(new_value, &global_var)); // 尝试写入新值,失败则重试__CLREX(); // 清除独占标记(防止后续LDREX受影响)
}int main(void) {atomic_write(0x12345678); // 原子写入新值while(1);
}