C/C++中整形读写的原子性

并发程序编写时,对单读单写的数据,通常认为是可以不用加锁的,最多注意下内存屏障;

问题描述

常见的单独单写并发时脏读的问题:

  • PA 同一地址的并发读写 POD类型(C8 / U8 / I16 / U16 / I32 / U32 / I64 / U64 / B8 / F32 / F64)
  • PB 同一地址的并发读写 SIMD 类型 (I128, U128)
  • PC 同一地址的并发读写 数组与结构体

分析

此部分代码是架构相关的,有Arm / X86 上迁移问题

  • A1 问题PA是基础,x86与Arm提供了不同的保证,以上基本类型内存对齐时,并发是可以保证的;
  • A2 问题P2,P3视为复杂类型写入和读取是多次操作,一定会脏读;

violate

C/C++ 中的 volatile - 知乎 (zhihu.com)

volatile 不能解决多线程中的问题。
按照 Hans Boehm & Nick Maclaren 的总结volatile 只在三种场合下是合适的。

  • 和信号处理(signal handler)相关的场合;
  • 和内存映射硬件(memory mapped hardware)相关的场合;
  • 和非本地跳转(setjmplongjmp)相关的场合。

Intel spec

Intel® 64 and IA-32 Architectures Software Developer Manuals 卷3 - 8.1.1

PA问题中,地址对齐的访问都是原子的,同一缓存行内和不触发 Page Fault 和 Cache Miss 的不对齐操作也是原子的。

Arm spec

Arm Architecture Reference Manual for A-profile architecture B2.2.1 & B2.5.2

PA问题中,地址对齐的访问是single - copy atomic,非地址对齐的访问在允许 SCTLR_ELx 且芯片支持 LSE,且数据在同一个16b以内是原子的。

内存对齐

  • 结构体中字段的偏移地址和结构体大小与编译器相关;
  • 栈变量和全局变量、静态变量的首地址与编译器相关;
  • 内存申请的首地址与运行时库和操作系统API相关;

gcc & msvc

(23条消息) 结构体字节对齐和位域对齐——VC、gcc_bytxl的博客-CSDN博客_gcc 结构体 起始地址
对齐应该确实与编译器相关,帖子中相当于 packed 属性默认值在 vs 和 gcc 中不一致,但是实际测试应该是都是 8。

linux malloc

malloc(3) - Linux manual page (man7.org)

The malloc() and calloc() functions return a pointer to the allocated memory, which is suitably aligned for any built-in type.

malloc.c - malloc/malloc.c - Glibc source code (glibc-2.37) - Bootlin 行 97

Alignment: 2 * sizeof(size_t) (default) (i.e., 8 byte alignment with 4byte size_t). This suffices for nearly all current machines and C compilers. However, you can define MALLOC_ALIGNMENT to be wider than this if necessary.

linux shmat & mmap

shmat(2): shared memory operations - Linux man page (die.net)

If shmaddr isn't NULL and SHM_RND is specified in shmflg, the attach occurs at the address equal to shmaddr rounded down to the nearest multiple of SHMLBA. Otherwise shmaddr must be a page-aligned address at which the attach occurs

自定义内存分配

自己做内存管理时,比如申请一大片内存,逐个切分时,会导致内存的起始地址不是对齐的,后续的字段也不对齐,从而潜在有并发问题;

PC 问题一些思考

写者持续更新 <a, b, c...>, 产生 <a1, b1, c1...> <a2, b2, c2....>多个版本的数据,读者持续读,预期是读到同一个版本的数据,不产生脏读,即读到 <a_i, b_i, c_i+1, ...>

加锁,如 spinlock

仿写 exanic-software/rwlock.h at master · cisco/exanic-software · GitHub

typedef uint32_t spinlock_t;
void spin_lock(spinlock_t* lock)
{
    uint16_t v2o = __sync_fetch_and_add((violate uint16_t*)lock+1, 1);
    while (*(violate uint16_t*)lock != v2o)
        __ia32_pause();
}

void spin_unlock(spinlock_t* lock)
{
    uint32_t v = *(violate uint32_t*)lock;
    if ((U16)v != (U16)(v >> 16))
        __sync_fetch_and_add((violate uint16_t*)lock, 1);
}

记录每个版本,更新版本号

typedef struct data_t {
    char a;
    char b;
    char c;
    char d;
} data_t

data_t datas[DATA_MAX];
uint32_t dataver;

Writer :                | Reader:
write datas[i]          | read dataver
write_barrier()         | read_barrier()  
write dataver           | write datas[i]

double check

typedef struct data_t {
    char a;
    char b;
    char c;
    char d;
} data_t

data_t data;
uint32_t dataver;

Writer :                | Reader:
write dataver           | 1 read dataver as dataver0
write_barrier()         | 2 read_barrier()  
write data              | 3 write data
                        | 4 if (daraver != dataver0) goto 1

总结

  • 地址对齐是认为,基本类型的读写是原子的,可跨平台的
  • violate 不是必须的
  • 更推荐用 CAS, FAA等原子操作显式地进行原子更新
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 216,039评论 6 498
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 92,223评论 3 392
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 161,916评论 0 351
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 58,009评论 1 291
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 67,030评论 6 388
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 51,011评论 1 295
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 39,934评论 3 416
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 38,754评论 0 271
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 45,202评论 1 309
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 37,433评论 2 331
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 39,590评论 1 346
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 35,321评论 5 342
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 40,917评论 3 325
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 31,568评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,738评论 1 268
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 47,583评论 2 368
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 44,482评论 2 352

推荐阅读更多精彩内容