stm32中优先使用原子操作的具体实现方式

在STM32中,优先使用的原子操作主要包括‌位带操作 ‌、‌LDREX/STREX指令 ‌以及‌CMSIS提供的原子操作宏‌,以下是具体实现方式‌12:


1. ‌**位带操作(Bit-Banding)**‌

通过硬件直接对单个比特进行原子读写,适用于GPIO或寄存器位操作‌1:

cpp 复制代码
#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:

cpp 复制代码
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_BITATOMIC_MODIFY_REG‌24:

cpp 复制代码
#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);
cpp 复制代码
#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:

cpp 复制代码
// 原子设置PB5高电平、PB6低电平
GPIOB->BSRR = (1 << 5) | (1 << (6 + 16));

临界区保护

HAL库提供__disable_irq()__enable_irq()函数,可临时禁用中断实现原子性‌

cpp 复制代码
volatile uint32_t counter = 0;

void atomic_increment(void) {
    __disable_irq();  // 进入临界区
    counter++;        // 原子操作
    __enable_irq();   // 退出临界区
}

代码实现示例

以下是一个综合示例,展示原子操作在多线程环境中的应用‌35:

atomic_ops.c

已完成

此代码演示了计数器原子递增和GPIO标志位原子设置,适用于中断与主程序共享数据的场景‌1

cpp 复制代码
#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;
}
cpp 复制代码
#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);
}
相关推荐
bai5459369 小时前
STM32 CubeIDE 通过PWM占空比控制舵机角度
stm32·单片机·嵌入式硬件
松涛和鸣11 小时前
72、IMX6ULL驱动实战:设备树(DTS/DTB)+ GPIO子系统+Platform总线
linux·服务器·arm开发·数据库·单片机
简单中的复杂11 小时前
【避坑指南】RK3576 Linux SDK 编译:解决 Buildroot 卡死在 host-gcc-final 的终极方案
linux·嵌入式硬件
上海合宙LuatOS11 小时前
LuatOS核心库API——【audio 】
java·网络·单片机·嵌入式硬件·物联网·音视频·硬件工程
Hhh __灏12 小时前
stm32的SRAM内存不足如何分析和优化?堆栈空间如何优化?
单片机
LS_learner12 小时前
Snapd和Apt—Linux 上两种完全不同的软件包管理系统
嵌入式硬件
点灯小铭12 小时前
基于51单片机的双档交流电压表设计与实现
单片机·嵌入式硬件·毕业设计·51单片机·课程设计·期末大作业
厦门辰迈智慧科技有限公司13 小时前
全功能数据采集仪mcu主要用途
单片机·嵌入式硬件·水库大坝安全监测·数据采集仪mcu
jl486382113 小时前
变比测试仪显示屏的“标杆“配置!如何兼顾30000小时寿命与六角矢量图精准显示?
人工智能·经验分享·嵌入式硬件·物联网·人机交互