c++ 嵌入汇编的方式实现int型自增

x86/x86_64 实现

x86 平台上,使用 LOCK XADD 指令来实现原子自增:

c++ 复制代码
#include <iostream>

inline int atomic_increment_x86(int* value) {
    int result;
    __asm__ __volatile__(
        "lock xaddl %1, %0"
        : "+m"(*value), "=r"(result)
        : "1"(1)
        : "memory"
    );
    return result + 1;
}

说明

  • lock xaddl 指令用于原子地将寄存器的值加到内存变量上,并返回原值。
  • +m(*value): +m 表示被修改的内存操作数。
  • "1"(1): 约束 1 号操作数使用寄存器,并初始化为 1
  • memory 作为 clobber 说明该指令会修改内存。

ARM(ARMv7)实现

在 ARM(32 位)上,使用 ldrexstrex 指令实现原子操作:

c++ 复制代码
#include <iostream>

inline int atomic_increment_arm(int* value) {
    int result, tmp;
    __asm__ __volatile__(
        "1: ldrex %0, [%2]\n"
        "   add %0, %0, #1\n"
        "   strex %1, %0, [%2]\n"
        "   teq %1, #0\n"
        "   bne 1b\n"
        : "=&r"(result), "=&r"(tmp)
        : "r"(value)
        : "memory", "cc"
    );
    return result;
}

说明

  • ldrex 加载值到 result,并设置独占标志。
  • add 执行加 1 操作。
  • strex 试图存回新值到 value,并检查是否成功(如果 strex 失败,则循环重试)。
  • teq %1, #0 检测 strex 失败标志,不为 0 时回到 1: 处重试。

ARM64(AArch64)实现

在 ARM64(64 位)上,可以使用 ldxrstxr 进行原子操作:

c++ 复制代码
#include <iostream>

inline int atomic_increment_arm64(int* value) {
    int result, tmp;
    __asm__ __volatile__(
        "1: ldxr %w0, [%2]\n"
        "   add %w0, %w0, #1\n"
        "   stxr %w1, %w0, [%2]\n"
        "   cbnz %w1, 1b\n"
        : "=&r"(result), "=&r"(tmp)
        : "r"(value)
        : "memory"
    );
    return result;
}

说明

  • ldxr(load exclusive register)加载 value,并建立独占访问。
  • stxr(store exclusive register)存储 value,如果失败,则重新加载并重试。
  • cbnz 指令检查 stxr 失败标志,不为 0 时回到 1: 处重试。

c++封装

c++ 复制代码
#include <iostream>

inline int atomic_increment(int* value) {
#if defined(__x86_64__) || defined(__i386__)
    return atomic_increment_x86(value);
#elif defined(__aarch64__)
    return atomic_increment_arm64(value);
#elif defined(__arm__)
    return atomic_increment_arm(value);
#else
    #error "Unsupported architecture"
#endif
}

int main() {
    int counter = 0;
    std::cout << "Before: " << counter << std::endl;
    std::cout << "After: " << atomic_increment(&counter) << std::endl;
    return 0;
}
相关推荐
じ☆ve 清风°3 分钟前
JavaScript 原型与原型链:深入理解 __proto__ 和 prototype 的由来与关系
开发语言·javascript·原型模式
_r0bin_5 小时前
前端面试准备-7
开发语言·前端·javascript·fetch·跨域·class
zhang98800005 小时前
JavaScript 核心原理深度解析-不停留于表面的VUE等的使用!
开发语言·javascript·vue.js
Fanxt_Ja6 小时前
【JVM】三色标记法原理
java·开发语言·jvm·算法
蓝婷儿6 小时前
6个月Python学习计划 Day 15 - 函数式编程、高阶函数、生成器/迭代器
开发语言·python·学习
love530love7 小时前
【笔记】在 MSYS2(MINGW64)中正确安装 Rust
运维·开发语言·人工智能·windows·笔记·python·rust
南郁7 小时前
007-nlohmann/json 项目应用-C++开源库108杰
c++·开源·json·nlohmann·现代c++·d2school·108杰
slandarer7 小时前
MATLAB | 绘图复刻(十九)| 轻松拿捏 Nature Communications 绘图
开发语言·matlab
狐凄7 小时前
Python实例题:Python计算二元二次方程组
开发语言·python
roman_日积跬步-终至千里8 小时前
【Go语言基础【3】】变量、常量、值类型与引用类型
开发语言·算法·golang