gcc_generic.h 5.6 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133
  1. /*
  2. Copyright 2005-2013 Intel Corporation. All Rights Reserved.
  3. This file is part of Threading Building Blocks.
  4. Threading Building Blocks is free software; you can redistribute it
  5. and/or modify it under the terms of the GNU General Public License
  6. version 2 as published by the Free Software Foundation.
  7. Threading Building Blocks is distributed in the hope that it will be
  8. useful, but WITHOUT ANY WARRANTY; without even the implied warranty
  9. of MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
  10. GNU General Public License for more details.
  11. You should have received a copy of the GNU General Public License
  12. along with Threading Building Blocks; if not, write to the Free Software
  13. Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA
  14. As a special exception, you may use this file as part of a free software
  15. library without restriction. Specifically, if other files instantiate
  16. templates or use macros or inline functions from this file, or you compile
  17. this file and link it with other files to produce an executable, this
  18. file does not by itself cause the resulting executable to be covered by
  19. the GNU General Public License. This exception does not however
  20. invalidate any other reasons why the executable file might be covered by
  21. the GNU General Public License.
  22. */
  23. #if !defined(__TBB_machine_H) || defined(__TBB_machine_gcc_generic_H)
  24. #error Do not #include this internal file directly; use public TBB headers instead.
  25. #endif
  26. #define __TBB_machine_gcc_generic_H
  27. #include <stdint.h>
  28. #include <unistd.h>
  29. #define __TBB_WORDSIZE __SIZEOF_POINTER__
  30. #if __TBB_GCC_64BIT_ATOMIC_BUILTINS_BROKEN
  31. #define __TBB_64BIT_ATOMICS 0
  32. #endif
  33. /** FPU control setting not available for non-Intel architectures on Android **/
  34. #if __ANDROID__ && __TBB_generic_arch
  35. #define __TBB_CPU_CTL_ENV_PRESENT 0
  36. #endif
  37. #ifdef __BYTE_ORDER__
  38. #if __BYTE_ORDER__==__ORDER_BIG_ENDIAN__
  39. #define __TBB_BIG_ENDIAN 1
  40. #elif __BYTE_ORDER__==__ORDER_LITTLE_ENDIAN__
  41. #define __TBB_BIG_ENDIAN 0
  42. #elif __BYTE_ORDER__==__ORDER_PDP_ENDIAN__
  43. #define __TBB_BIG_ENDIAN -1 // not currently supported
  44. #endif
  45. #endif
  46. /** As this generic implementation has absolutely no information about underlying
  47. hardware, its performance most likely will be sub-optimal because of full memory
  48. fence usages where a more lightweight synchronization means (or none at all)
  49. could suffice. Thus if you use this header to enable TBB on a new platform,
  50. consider forking it and relaxing below helpers as appropriate. **/
  51. #define __TBB_acquire_consistency_helper() __sync_synchronize()
  52. #define __TBB_release_consistency_helper() __sync_synchronize()
  53. #define __TBB_full_memory_fence() __sync_synchronize()
  54. #define __TBB_control_consistency_helper() __sync_synchronize()
  55. #define __TBB_MACHINE_DEFINE_ATOMICS(S,T) \
  56. inline T __TBB_machine_cmpswp##S( volatile void *ptr, T value, T comparand ) { \
  57. return __sync_val_compare_and_swap(reinterpret_cast<volatile T *>(ptr),comparand,value); \
  58. } \
  59. \
  60. inline T __TBB_machine_fetchadd##S( volatile void *ptr, T value ) { \
  61. return __sync_fetch_and_add(reinterpret_cast<volatile T *>(ptr),value); \
  62. } \
  63. __TBB_MACHINE_DEFINE_ATOMICS(1,int8_t)
  64. __TBB_MACHINE_DEFINE_ATOMICS(2,int16_t)
  65. __TBB_MACHINE_DEFINE_ATOMICS(4,int32_t)
  66. __TBB_MACHINE_DEFINE_ATOMICS(8,int64_t)
  67. #undef __TBB_MACHINE_DEFINE_ATOMICS
  68. namespace tbb{ namespace internal { namespace gcc_builtins {
  69. inline int clz(unsigned int x){ return __builtin_clz(x);};
  70. inline int clz(unsigned long int x){ return __builtin_clzl(x);};
  71. inline int clz(unsigned long long int x){ return __builtin_clzll(x);};
  72. }}}
  73. //gcc __builtin_clz builtin count _number_ of leading zeroes
  74. static inline intptr_t __TBB_machine_lg( uintptr_t x ) {
  75. return sizeof(x)*8 - tbb::internal::gcc_builtins::clz(x) -1 ;
  76. }
  77. static inline void __TBB_machine_or( volatile void *ptr, uintptr_t addend ) {
  78. __sync_fetch_and_or(reinterpret_cast<volatile uintptr_t *>(ptr),addend);
  79. }
  80. static inline void __TBB_machine_and( volatile void *ptr, uintptr_t addend ) {
  81. __sync_fetch_and_and(reinterpret_cast<volatile uintptr_t *>(ptr),addend);
  82. }
  83. typedef unsigned char __TBB_Flag;
  84. typedef __TBB_atomic __TBB_Flag __TBB_atomic_flag;
  85. inline bool __TBB_machine_try_lock_byte( __TBB_atomic_flag &flag ) {
  86. return __sync_lock_test_and_set(&flag,1)==0;
  87. }
  88. inline void __TBB_machine_unlock_byte( __TBB_atomic_flag &flag , __TBB_Flag) {
  89. __sync_lock_release(&flag);
  90. }
  91. // Machine specific atomic operations
  92. #define __TBB_AtomicOR(P,V) __TBB_machine_or(P,V)
  93. #define __TBB_AtomicAND(P,V) __TBB_machine_and(P,V)
  94. #define __TBB_TryLockByte __TBB_machine_try_lock_byte
  95. #define __TBB_UnlockByte __TBB_machine_unlock_byte
  96. // Definition of other functions
  97. #define __TBB_Log2(V) __TBB_machine_lg(V)
  98. #define __TBB_USE_GENERIC_FETCH_STORE 1
  99. #define __TBB_USE_GENERIC_HALF_FENCED_LOAD_STORE 1
  100. #define __TBB_USE_GENERIC_RELAXED_LOAD_STORE 1
  101. #define __TBB_USE_GENERIC_SEQUENTIAL_CONSISTENCY_LOAD_STORE 1
  102. #if __TBB_WORDSIZE==4
  103. #define __TBB_USE_GENERIC_DWORD_LOAD_STORE 1
  104. #endif