compiler.h 4.3 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127
  1. #pragma once
  2. #define __force __attribute__((force))
  3. #define likely(x) __builtin_expect(!!(x), 1)
  4. #define unlikely(x) __builtin_expect(!!(x), 0)
  5. #ifndef barrier
  6. // 内存屏障
  7. #define barrier() __asm__ __volatile__("" :: \
  8. : "memory");
  9. #endif
  10. // 编译器属性
  11. // 当函数的返回值未被使用时,编译器抛出警告信息
  12. #define __must_check __attribute__((__warn_unused_result__))
  13. typedef uint8_t __attribute__((__may_alias__)) __u8_alias_t;
  14. typedef uint16_t __attribute__((__may_alias__)) __u16_alias_t;
  15. typedef uint32_t __attribute__((__may_alias__)) __u32_alias_t;
  16. typedef uint64_t __attribute__((__may_alias__)) __u64_alias_t;
  17. /**
  18. * @brief 从src读取数据到dst,该过程避免编译器优化。
  19. *
  20. * @param dst 目标地址指针
  21. * @param src 源地址指针
  22. * @param size 要读取的数据大小(建议1、2、4、8字节,若不满足要求,则采用memcpy读取。)
  23. */
  24. static __always_inline void __read_once_size(void *dst, const volatile void *src, int size)
  25. {
  26. switch (size)
  27. {
  28. case 1:
  29. *(__u8_alias_t *)dst = *(volatile __u8_alias_t *)src;
  30. break;
  31. case 2:
  32. *(__u16_alias_t *)dst = *(volatile __u16_alias_t *)src;
  33. break;
  34. case 4:
  35. *(__u32_alias_t *)dst = *(volatile __u32_alias_t *)src;
  36. break;
  37. case 8:
  38. *(__u64_alias_t *)dst = *(volatile __u64_alias_t *)src;
  39. break;
  40. default:
  41. barrier();
  42. __builtin_memcpy((void *)dst, (const void *)src, size);
  43. barrier();
  44. break;
  45. }
  46. }
  47. /**
  48. * @brief 把src处的数据到dst,该过程避免编译器优化。
  49. *
  50. * @param dst 目标地址指针
  51. * @param src 源地址指针
  52. * @param size 要写入的数据大小(建议1、2、4、8字节,若不满足要求,则采用memcpy传输。)
  53. */
  54. static __always_inline void __write_once_size(volatile void *dst, void *src, int size)
  55. {
  56. switch (size)
  57. {
  58. case 1:
  59. *(volatile __u8_alias_t *)dst = *(__u8_alias_t *)src;
  60. break;
  61. case 2:
  62. *(volatile __u16_alias_t *)dst = *(__u16_alias_t *)src;
  63. break;
  64. case 4:
  65. *(volatile __u32_alias_t *)dst = *(__u32_alias_t *)src;
  66. break;
  67. case 8:
  68. *(volatile __u64_alias_t *)dst = *(__u64_alias_t *)src;
  69. break;
  70. default:
  71. barrier();
  72. __builtin_memcpy((void *)dst, (const void *)src, size);
  73. barrier();
  74. break;
  75. }
  76. }
  77. /**
  78. * 这两个宏能够避免编译器重排序、合并涉及到的读写操作,从而避免由于编译器优化导致的多线程读写顺序错误。
  79. * 通过将有顺序要求的两个读/写操作放置在READ_ONCE()和WRITE_ONCE()之中,能够让编译器知道这些操作具有顺序要求。
  80. *
  81. * 这两个宏同样适用于Union或struct。如果要访问的数据大小不是1、2、4、8字节,则会使用memcpy来处理。
  82. *
  83. * 这两个宏的主要使用场景:
  84. * 1.两个进程或者中断处理函数之间的信息交流与沟通
  85. * 2.确保编译器不会折叠、旋转或以其他方式对代码进行优化,从而破坏数据访问顺序。
  86. *
  87. * 这两个宏的union __u内的__c用作这个union的地址的指针
  88. *
  89. * 关于READ_ONCE和WRITE_ONCE的简单说明,请转到:https://bbs.dragonos.org/forum.php?mod=viewthread&tid=24
  90. */
  91. /**
  92. * @brief 读取变量x (避免编译器优化)
  93. */
  94. #define READ_ONCE(x) \
  95. ({ \
  96. union \
  97. { \
  98. typeof(x) __val; \
  99. char __c[1]; \
  100. } __u = {.__c = {0}}; \
  101. __read_once_size(__u.__c, &(x), sizeof(x)); \
  102. __u.__val; \
  103. })
  104. /**
  105. * @brief 将val写入变量x (避免编译器优化)
  106. */
  107. #define WRITE_ONCE(x, val) \
  108. ({ \
  109. union \
  110. { \
  111. typeof(x) __val; \
  112. char __c[1]; \
  113. } __u = {.val = (val)}; \
  114. __write_once_size(&(x), __u.__c, sizeof(x)); \
  115. __u.__val; \
  116. })