1 #pragma once
2
3 #include <common/compiler_attributes.h>
4
5 #define likely(x) __builtin_expect(!!(x), 1)
6 #define unlikely(x) __builtin_expect(!!(x), 0)
7
8 #ifndef barrier
9 // 内存屏障
10 #define barrier() __asm__ __volatile__("" :: \
11 : "memory");
12 #endif
13
14 /**
15 * @brief 编译时断言,如果condition不为1,则输出msg
16 *
17 * @param prefix 一个“不存在的函数名”的前缀
18 * @param suffix 一个“不存在的函数名”的后缀
19 */
20 #define __compiletime_assert(condition, msg, prefix, suffix) \
21 do \
22 { \
23 /** \
24 * 声明一个不存在的函数的extern,如果assert失败,就调用它,从而导致 \
25 * 链接时出错,进而达到“编译时断言”的功能。 \
26 */ \
27 __noreturn extern void prefix##suffix(void) \
28 __compiletime_error(msg); \
29 if (!(condition)) \
30 prefix##suffix(); \
31 } while (0)
32
33 /**
34 * @brief 当condition是false时,中断编译,并输出指定的错误信息
35 *
36 * @param condition assert的情况
37 * @param msg condition为false时输出的错误信息
38 */
39 #define complietime_assert(condition, msg) \
40 __compiletime_assert(condition, msg, __compiletime_assert__, __COUNTER__)
41
42 /**
43 * @brief 从src读取数据到dst,该过程避免编译器优化。
44 *
45 * @param dst 目标地址指针
46 * @param src 源地址指针
47 * @param size 要读取的数据大小(建议1、2、4、8字节,若不满足要求,则采用memcpy读取。)
48 */
__read_once_size(void * dst,const volatile void * src,int size)49 static __always_inline void __read_once_size(void *dst, const volatile void *src, int size)
50 {
51 switch (size)
52 {
53 case 1:
54 *(__u8_alias_t *)dst = *(volatile __u8_alias_t *)src;
55 break;
56 case 2:
57 *(__u16_alias_t *)dst = *(volatile __u16_alias_t *)src;
58 break;
59 case 4:
60 *(__u32_alias_t *)dst = *(volatile __u32_alias_t *)src;
61 break;
62 case 8:
63 *(__u64_alias_t *)dst = *(volatile __u64_alias_t *)src;
64 break;
65 default:
66 barrier();
67 __builtin_memcpy((void *)dst, (const void *)src, size);
68 barrier();
69 break;
70 }
71 }
72
73 /**
74 * @brief 把src处的数据到dst,该过程避免编译器优化。
75 *
76 * @param dst 目标地址指针
77 * @param src 源地址指针
78 * @param size 要写入的数据大小(建议1、2、4、8字节,若不满足要求,则采用memcpy传输。)
79 */
__write_once_size(volatile void * dst,void * src,int size)80 static __always_inline void __write_once_size(volatile void *dst, void *src, int size)
81 {
82 switch (size)
83 {
84 case 1:
85 *(volatile __u8_alias_t *)dst = *(__u8_alias_t *)src;
86 break;
87 case 2:
88 *(volatile __u16_alias_t *)dst = *(__u16_alias_t *)src;
89 break;
90 case 4:
91 *(volatile __u32_alias_t *)dst = *(__u32_alias_t *)src;
92 break;
93 case 8:
94 *(volatile __u64_alias_t *)dst = *(__u64_alias_t *)src;
95 break;
96 default:
97 barrier();
98 __builtin_memcpy((void *)dst, (const void *)src, size);
99 barrier();
100 break;
101 }
102 }
103
104 /**
105 * 这两个宏能够避免编译器重排序、合并涉及到的读写操作,从而避免由于编译器优化导致的多线程读写顺序错误。
106 * 通过将有顺序要求的两个读/写操作放置在READ_ONCE()和WRITE_ONCE()之中,能够让编译器知道这些操作具有顺序要求。
107 *
108 * 这两个宏同样适用于Union或struct。如果要访问的数据大小不是1、2、4、8字节,则会使用memcpy来处理。
109 *
110 * 这两个宏的主要使用场景:
111 * 1.两个进程或者中断处理函数之间的信息交流与沟通
112 * 2.确保编译器不会折叠、旋转或以其他方式对代码进行优化,从而破坏数据访问顺序。
113 *
114 * 这两个宏的union __u内的__c用作这个union的地址的指针
115 *
116 * 关于READ_ONCE和WRITE_ONCE的简单说明,请转到:https://bbs.dragonos.org/forum.php?mod=viewthread&tid=24
117 */
118
119 /**
120 * @brief 读取变量x (避免编译器优化)
121 */
122 #define READ_ONCE(x) \
123 ({ \
124 union \
125 { \
126 typeof(x) __val; \
127 char __c[1]; \
128 } __u = {.__c = {0}}; \
129 __read_once_size(__u.__c, &(x), sizeof(x)); \
130 __u.__val; \
131 })
132
133 /**
134 * @brief 将val写入变量x (避免编译器优化)
135 */
136 #define WRITE_ONCE(x, val) \
137 ({ \
138 union \
139 { \
140 typeof(x) __val; \
141 char __c[1]; \
142 } __u = {.val = (val)}; \
143 __write_once_size(&(x), __u.__c, sizeof(x)); \
144 __u.__val; \
145 })
146