当前位置: 首页 > news >正文

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_BITATOMIC_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);
}

http://www.dtcms.com/a/307625.html

相关文章:

  • leecode611 有效三角形的个数
  • 基于N32G45x+RTT驱动框架的定时器外部计数
  • WebMvcConfigurer配置接口详解
  • ClickHouse vs PostgreSQL:数据分析领域的王者之争,谁更胜一筹?
  • 模型优化——在MacOS 上使用 Python 脚本批量大幅度精简 GLB 模型(通过 Blender 处理)
  • 【linux驱动开发】Vscode + Remote SSH + clangd + bear=内核源码阅读环境搭建
  • Visual Studio Code (VSCode) 的常用快捷键
  • 33.【.NET8 实战--孢子记账--从单体到微服务--转向微服务】--单体转微服务--财务服务--记账
  • Shader开发(五)什么是渲染管线
  • 【大模型理论篇】混合思考之自适应思维链
  • day28_2025-07-31
  • 基于京东评论的文本挖掘与分析,使用LSTM情感分析算法以及网络语义分析
  • 【数据结构】算法代码
  • 前端框架Vue3(三)——路由和pinia
  • 分布内侧内嗅皮层的层Ⅱ或层Ⅲ的网格细胞(grid cells)对NLP中的深层语义分析的积极影响和启示
  • vue3.0 +TypeScript 项目中pinia基础语法和使用
  • 【大数据】open_metadata 开源元数据管理平台建设与数据血缘实践
  • 「源力觉醒 创作者计划」开源大模型重构数智文明新范式
  • AI任务相关解决方案12-NLP的15项任务大融合系统:传统NLP与Qwen大模型的深度结合
  • NTLDR源代码分析之从GetSector函数到blread函数
  • 解决 IntelliJ IDEA Build时 Lombok 不生效问题
  • 商旅平台怎么选?如何规避商旅流程中的违规风险?
  • 【未解决】STM32无刷电机驱动电路问题记录
  • .NET Core部署服务器
  • 智慧收银系统开发进销存库存统计,便利店、水果店、建材与家居行业的库存汇总管理—仙盟创梦IDE
  • Spring Boot 异常处理:从全局捕获到优化用户体验!
  • PostgreSQL面试题及详细答案120道(01-20)
  • 解放双手!Report Distro 实现报表自动化分发
  • 微软发布Microsoft Sentinel数据湖国际版
  • SecurityContextHolder 管理安全上下文的核心组件详解