did_you_mean 0.9.0 → 0.9.2

Sign up to get free protection for your applications and to get access to all the features.
Files changed (205) hide show
  1. checksums.yaml +4 -4
  2. data/.gitignore +2 -0
  3. data/.travis.yml +1 -1
  4. data/CHANGELOG.md +150 -0
  5. data/README.md +14 -1
  6. data/did_you_mean.gemspec +0 -1
  7. data/ext/did_you_mean/extconf.rb +14 -11
  8. data/ext/did_you_mean/ruby_headers/215/id.h +173 -0
  9. data/ext/did_you_mean/ruby_headers/215/internal.h +892 -0
  10. data/ext/did_you_mean/ruby_headers/215/method.h +142 -0
  11. data/ext/did_you_mean/ruby_headers/215/node.h +543 -0
  12. data/ext/did_you_mean/ruby_headers/215/ruby_atomic.h +170 -0
  13. data/ext/did_you_mean/ruby_headers/215/thread_native.h +23 -0
  14. data/ext/did_you_mean/ruby_headers/215/thread_pthread.h +56 -0
  15. data/ext/did_you_mean/ruby_headers/215/thread_win32.h +45 -0
  16. data/ext/did_you_mean/ruby_headers/215/vm_core.h +1055 -0
  17. data/ext/did_you_mean/ruby_headers/215/vm_debug.h +37 -0
  18. data/ext/did_you_mean/ruby_headers/215/vm_opts.h +56 -0
  19. data/ext/did_you_mean/ruby_headers/220/internal.h +11 -6
  20. data/ext/did_you_mean/ruby_headers/220/vm_core.h +1 -0
  21. data/lib/did_you_mean/finders.rb +1 -1
  22. data/lib/did_you_mean/test_helper.rb +0 -4
  23. data/lib/did_you_mean/version.rb +1 -1
  24. data/test/no_method_error_extension_test.rb +0 -6
  25. data/test/test_helper.rb +0 -1
  26. metadata +18 -196
  27. data/docs/did_you_mean_example.png +0 -0
  28. data/ext/did_you_mean/ruby_headers/193/addr2line.h +0 -21
  29. data/ext/did_you_mean/ruby_headers/193/constant.h +0 -34
  30. data/ext/did_you_mean/ruby_headers/193/dln.h +0 -50
  31. data/ext/did_you_mean/ruby_headers/193/encdb.h +0 -170
  32. data/ext/did_you_mean/ruby_headers/193/eval_intern.h +0 -234
  33. data/ext/did_you_mean/ruby_headers/193/gc.h +0 -99
  34. data/ext/did_you_mean/ruby_headers/193/iseq.h +0 -126
  35. data/ext/did_you_mean/ruby_headers/193/parse.h +0 -183
  36. data/ext/did_you_mean/ruby_headers/193/probes.h +0 -83
  37. data/ext/did_you_mean/ruby_headers/193/regenc.h +0 -219
  38. data/ext/did_you_mean/ruby_headers/193/regint.h +0 -850
  39. data/ext/did_you_mean/ruby_headers/193/regparse.h +0 -362
  40. data/ext/did_you_mean/ruby_headers/193/revision.h +0 -1
  41. data/ext/did_you_mean/ruby_headers/193/siphash.h +0 -48
  42. data/ext/did_you_mean/ruby_headers/193/timev.h +0 -21
  43. data/ext/did_you_mean/ruby_headers/193/transcode_data.h +0 -117
  44. data/ext/did_you_mean/ruby_headers/193/transdb.h +0 -190
  45. data/ext/did_you_mean/ruby_headers/193/verconf.h +0 -13
  46. data/ext/did_you_mean/ruby_headers/193/version.h +0 -52
  47. data/ext/did_you_mean/ruby_headers/193/vm_exec.h +0 -184
  48. data/ext/did_you_mean/ruby_headers/193/vm_insnhelper.h +0 -220
  49. data/ext/did_you_mean/ruby_headers/200/addr2line.h +0 -21
  50. data/ext/did_you_mean/ruby_headers/200/constant.h +0 -36
  51. data/ext/did_you_mean/ruby_headers/200/dln.h +0 -50
  52. data/ext/did_you_mean/ruby_headers/200/encdb.h +0 -170
  53. data/ext/did_you_mean/ruby_headers/200/eval_intern.h +0 -213
  54. data/ext/did_you_mean/ruby_headers/200/gc.h +0 -104
  55. data/ext/did_you_mean/ruby_headers/200/iseq.h +0 -140
  56. data/ext/did_you_mean/ruby_headers/200/parse.h +0 -183
  57. data/ext/did_you_mean/ruby_headers/200/probes.h +0 -83
  58. data/ext/did_you_mean/ruby_headers/200/probes_helper.h +0 -67
  59. data/ext/did_you_mean/ruby_headers/200/regenc.h +0 -227
  60. data/ext/did_you_mean/ruby_headers/200/regint.h +0 -915
  61. data/ext/did_you_mean/ruby_headers/200/regparse.h +0 -367
  62. data/ext/did_you_mean/ruby_headers/200/revision.h +0 -1
  63. data/ext/did_you_mean/ruby_headers/200/siphash.h +0 -48
  64. data/ext/did_you_mean/ruby_headers/200/timev.h +0 -21
  65. data/ext/did_you_mean/ruby_headers/200/transcode_data.h +0 -127
  66. data/ext/did_you_mean/ruby_headers/200/transdb.h +0 -190
  67. data/ext/did_you_mean/ruby_headers/200/verconf.h +0 -13
  68. data/ext/did_you_mean/ruby_headers/200/version.h +0 -52
  69. data/ext/did_you_mean/ruby_headers/200/vm_exec.h +0 -173
  70. data/ext/did_you_mean/ruby_headers/200/vm_insnhelper.h +0 -274
  71. data/ext/did_you_mean/ruby_headers/210/addr2line.h +0 -21
  72. data/ext/did_you_mean/ruby_headers/210/constant.h +0 -36
  73. data/ext/did_you_mean/ruby_headers/210/dln.h +0 -51
  74. data/ext/did_you_mean/ruby_headers/210/encdb.h +0 -170
  75. data/ext/did_you_mean/ruby_headers/210/eval_intern.h +0 -260
  76. data/ext/did_you_mean/ruby_headers/210/gc.h +0 -101
  77. data/ext/did_you_mean/ruby_headers/210/iseq.h +0 -136
  78. data/ext/did_you_mean/ruby_headers/210/parse.h +0 -183
  79. data/ext/did_you_mean/ruby_headers/210/probes.h +0 -83
  80. data/ext/did_you_mean/ruby_headers/210/probes_helper.h +0 -67
  81. data/ext/did_you_mean/ruby_headers/210/regenc.h +0 -223
  82. data/ext/did_you_mean/ruby_headers/210/regint.h +0 -911
  83. data/ext/did_you_mean/ruby_headers/210/regparse.h +0 -363
  84. data/ext/did_you_mean/ruby_headers/210/revision.h +0 -1
  85. data/ext/did_you_mean/ruby_headers/210/siphash.h +0 -48
  86. data/ext/did_you_mean/ruby_headers/210/timev.h +0 -42
  87. data/ext/did_you_mean/ruby_headers/210/transcode_data.h +0 -123
  88. data/ext/did_you_mean/ruby_headers/210/transdb.h +0 -190
  89. data/ext/did_you_mean/ruby_headers/210/verconf.h +0 -13
  90. data/ext/did_you_mean/ruby_headers/210/version.h +0 -52
  91. data/ext/did_you_mean/ruby_headers/210/vm_exec.h +0 -182
  92. data/ext/did_you_mean/ruby_headers/210/vm_insnhelper.h +0 -273
  93. data/ext/did_you_mean/ruby_headers/211/addr2line.h +0 -21
  94. data/ext/did_you_mean/ruby_headers/211/constant.h +0 -36
  95. data/ext/did_you_mean/ruby_headers/211/dln.h +0 -51
  96. data/ext/did_you_mean/ruby_headers/211/encdb.h +0 -170
  97. data/ext/did_you_mean/ruby_headers/211/eval_intern.h +0 -260
  98. data/ext/did_you_mean/ruby_headers/211/gc.h +0 -101
  99. data/ext/did_you_mean/ruby_headers/211/iseq.h +0 -136
  100. data/ext/did_you_mean/ruby_headers/211/parse.h +0 -183
  101. data/ext/did_you_mean/ruby_headers/211/probes.h +0 -83
  102. data/ext/did_you_mean/ruby_headers/211/probes_helper.h +0 -67
  103. data/ext/did_you_mean/ruby_headers/211/regenc.h +0 -223
  104. data/ext/did_you_mean/ruby_headers/211/regint.h +0 -911
  105. data/ext/did_you_mean/ruby_headers/211/regparse.h +0 -363
  106. data/ext/did_you_mean/ruby_headers/211/revision.h +0 -1
  107. data/ext/did_you_mean/ruby_headers/211/siphash.h +0 -48
  108. data/ext/did_you_mean/ruby_headers/211/timev.h +0 -42
  109. data/ext/did_you_mean/ruby_headers/211/transcode_data.h +0 -123
  110. data/ext/did_you_mean/ruby_headers/211/transdb.h +0 -190
  111. data/ext/did_you_mean/ruby_headers/211/verconf.h +0 -13
  112. data/ext/did_you_mean/ruby_headers/211/version.h +0 -52
  113. data/ext/did_you_mean/ruby_headers/211/vm_exec.h +0 -182
  114. data/ext/did_you_mean/ruby_headers/211/vm_insnhelper.h +0 -273
  115. data/ext/did_you_mean/ruby_headers/212/addr2line.h +0 -21
  116. data/ext/did_you_mean/ruby_headers/212/constant.h +0 -36
  117. data/ext/did_you_mean/ruby_headers/212/dln.h +0 -51
  118. data/ext/did_you_mean/ruby_headers/212/encdb.h +0 -170
  119. data/ext/did_you_mean/ruby_headers/212/eval_intern.h +0 -260
  120. data/ext/did_you_mean/ruby_headers/212/gc.h +0 -101
  121. data/ext/did_you_mean/ruby_headers/212/iseq.h +0 -136
  122. data/ext/did_you_mean/ruby_headers/212/parse.h +0 -183
  123. data/ext/did_you_mean/ruby_headers/212/probes.h +0 -83
  124. data/ext/did_you_mean/ruby_headers/212/probes_helper.h +0 -67
  125. data/ext/did_you_mean/ruby_headers/212/regenc.h +0 -223
  126. data/ext/did_you_mean/ruby_headers/212/regint.h +0 -911
  127. data/ext/did_you_mean/ruby_headers/212/regparse.h +0 -363
  128. data/ext/did_you_mean/ruby_headers/212/revision.h +0 -1
  129. data/ext/did_you_mean/ruby_headers/212/siphash.h +0 -48
  130. data/ext/did_you_mean/ruby_headers/212/timev.h +0 -42
  131. data/ext/did_you_mean/ruby_headers/212/transcode_data.h +0 -123
  132. data/ext/did_you_mean/ruby_headers/212/transdb.h +0 -190
  133. data/ext/did_you_mean/ruby_headers/212/verconf.h +0 -13
  134. data/ext/did_you_mean/ruby_headers/212/version.h +0 -52
  135. data/ext/did_you_mean/ruby_headers/212/vm_exec.h +0 -182
  136. data/ext/did_you_mean/ruby_headers/212/vm_insnhelper.h +0 -273
  137. data/ext/did_you_mean/ruby_headers/213/addr2line.h +0 -21
  138. data/ext/did_you_mean/ruby_headers/213/constant.h +0 -36
  139. data/ext/did_you_mean/ruby_headers/213/dln.h +0 -51
  140. data/ext/did_you_mean/ruby_headers/213/encdb.h +0 -170
  141. data/ext/did_you_mean/ruby_headers/213/eval_intern.h +0 -270
  142. data/ext/did_you_mean/ruby_headers/213/gc.h +0 -101
  143. data/ext/did_you_mean/ruby_headers/213/iseq.h +0 -136
  144. data/ext/did_you_mean/ruby_headers/213/parse.h +0 -183
  145. data/ext/did_you_mean/ruby_headers/213/probes.h +0 -83
  146. data/ext/did_you_mean/ruby_headers/213/probes_helper.h +0 -67
  147. data/ext/did_you_mean/ruby_headers/213/regenc.h +0 -223
  148. data/ext/did_you_mean/ruby_headers/213/regint.h +0 -911
  149. data/ext/did_you_mean/ruby_headers/213/regparse.h +0 -363
  150. data/ext/did_you_mean/ruby_headers/213/revision.h +0 -1
  151. data/ext/did_you_mean/ruby_headers/213/siphash.h +0 -48
  152. data/ext/did_you_mean/ruby_headers/213/timev.h +0 -42
  153. data/ext/did_you_mean/ruby_headers/213/transcode_data.h +0 -123
  154. data/ext/did_you_mean/ruby_headers/213/transdb.h +0 -190
  155. data/ext/did_you_mean/ruby_headers/213/verconf.h +0 -13
  156. data/ext/did_you_mean/ruby_headers/213/version.h +0 -52
  157. data/ext/did_you_mean/ruby_headers/213/vm_exec.h +0 -182
  158. data/ext/did_you_mean/ruby_headers/213/vm_insnhelper.h +0 -273
  159. data/ext/did_you_mean/ruby_headers/214/addr2line.h +0 -21
  160. data/ext/did_you_mean/ruby_headers/214/constant.h +0 -36
  161. data/ext/did_you_mean/ruby_headers/214/dln.h +0 -51
  162. data/ext/did_you_mean/ruby_headers/214/encdb.h +0 -170
  163. data/ext/did_you_mean/ruby_headers/214/eval_intern.h +0 -270
  164. data/ext/did_you_mean/ruby_headers/214/gc.h +0 -101
  165. data/ext/did_you_mean/ruby_headers/214/iseq.h +0 -136
  166. data/ext/did_you_mean/ruby_headers/214/parse.h +0 -183
  167. data/ext/did_you_mean/ruby_headers/214/probes.h +0 -83
  168. data/ext/did_you_mean/ruby_headers/214/probes_helper.h +0 -67
  169. data/ext/did_you_mean/ruby_headers/214/regenc.h +0 -223
  170. data/ext/did_you_mean/ruby_headers/214/regint.h +0 -911
  171. data/ext/did_you_mean/ruby_headers/214/regparse.h +0 -363
  172. data/ext/did_you_mean/ruby_headers/214/revision.h +0 -1
  173. data/ext/did_you_mean/ruby_headers/214/siphash.h +0 -48
  174. data/ext/did_you_mean/ruby_headers/214/timev.h +0 -42
  175. data/ext/did_you_mean/ruby_headers/214/transcode_data.h +0 -123
  176. data/ext/did_you_mean/ruby_headers/214/transdb.h +0 -190
  177. data/ext/did_you_mean/ruby_headers/214/verconf.h +0 -13
  178. data/ext/did_you_mean/ruby_headers/214/version.h +0 -52
  179. data/ext/did_you_mean/ruby_headers/214/vm_exec.h +0 -182
  180. data/ext/did_you_mean/ruby_headers/214/vm_insnhelper.h +0 -273
  181. data/ext/did_you_mean/ruby_headers/220/addr2line.h +0 -21
  182. data/ext/did_you_mean/ruby_headers/220/ccan/build_assert/build_assert.h +0 -40
  183. data/ext/did_you_mean/ruby_headers/220/ccan/licenses/BSD-MIT +0 -17
  184. data/ext/did_you_mean/ruby_headers/220/ccan/licenses/CC0 +0 -28
  185. data/ext/did_you_mean/ruby_headers/220/constant.h +0 -43
  186. data/ext/did_you_mean/ruby_headers/220/dln.h +0 -51
  187. data/ext/did_you_mean/ruby_headers/220/encdb.h +0 -170
  188. data/ext/did_you_mean/ruby_headers/220/eval_intern.h +0 -275
  189. data/ext/did_you_mean/ruby_headers/220/gc.h +0 -107
  190. data/ext/did_you_mean/ruby_headers/220/iseq.h +0 -157
  191. data/ext/did_you_mean/ruby_headers/220/parse.h +0 -183
  192. data/ext/did_you_mean/ruby_headers/220/probes.h +0 -83
  193. data/ext/did_you_mean/ruby_headers/220/probes_helper.h +0 -67
  194. data/ext/did_you_mean/ruby_headers/220/regenc.h +0 -234
  195. data/ext/did_you_mean/ruby_headers/220/regint.h +0 -972
  196. data/ext/did_you_mean/ruby_headers/220/regparse.h +0 -363
  197. data/ext/did_you_mean/ruby_headers/220/revision.h +0 -1
  198. data/ext/did_you_mean/ruby_headers/220/siphash.h +0 -48
  199. data/ext/did_you_mean/ruby_headers/220/timev.h +0 -42
  200. data/ext/did_you_mean/ruby_headers/220/transcode_data.h +0 -123
  201. data/ext/did_you_mean/ruby_headers/220/transdb.h +0 -190
  202. data/ext/did_you_mean/ruby_headers/220/verconf.h +0 -13
  203. data/ext/did_you_mean/ruby_headers/220/version.h +0 -53
  204. data/ext/did_you_mean/ruby_headers/220/vm_exec.h +0 -182
  205. data/ext/did_you_mean/ruby_headers/220/vm_insnhelper.h +0 -233
@@ -0,0 +1,170 @@
1
+ #ifndef RUBY_ATOMIC_H
2
+ #define RUBY_ATOMIC_H
3
+
4
+ #if 0
5
+ #elif defined HAVE_GCC_ATOMIC_BUILTINS
6
+ typedef unsigned int rb_atomic_t;
7
+ # define ATOMIC_SET(var, val) (void)__atomic_exchange_n(&(var), (val), __ATOMIC_SEQ_CST)
8
+ # define ATOMIC_INC(var) __atomic_fetch_add(&(var), 1, __ATOMIC_SEQ_CST)
9
+ # define ATOMIC_DEC(var) __atomic_fetch_sub(&(var), 1, __ATOMIC_SEQ_CST)
10
+ # define ATOMIC_OR(var, val) __atomic_or_fetch(&(var), (val), __ATOMIC_SEQ_CST)
11
+ # define ATOMIC_EXCHANGE(var, val) __atomic_exchange_n(&(var), (val), __ATOMIC_SEQ_CST)
12
+ # define ATOMIC_CAS(var, oldval, newval) \
13
+ ({ __typeof__(oldval) oldvaldup = (oldval); /* oldval should not be modified */ \
14
+ __atomic_compare_exchange_n(&(var), &oldvaldup, (newval), 0, __ATOMIC_SEQ_CST, __ATOMIC_SEQ_CST); \
15
+ oldvaldup; })
16
+
17
+ # define ATOMIC_SIZE_ADD(var, val) __atomic_fetch_add(&(var), (val), __ATOMIC_SEQ_CST)
18
+ # define ATOMIC_SIZE_SUB(var, val) __atomic_fetch_sub(&(var), (val), __ATOMIC_SEQ_CST)
19
+
20
+ # define ATOMIC_PTR_EXCHANGE(var, val) __atomic_exchange_n(&(var), (val), __ATOMIC_SEQ_CST)
21
+
22
+ #elif defined HAVE_GCC_SYNC_BUILTINS
23
+ /* @shyouhei hack to support atomic operations in case of gcc. Gcc
24
+ * has its own pseudo-insns to support them. See info, or
25
+ * http://gcc.gnu.org/onlinedocs/gcc/Atomic-Builtins.html */
26
+
27
+ typedef unsigned int rb_atomic_t; /* Anything OK */
28
+ # define ATOMIC_SET(var, val) (void)__sync_lock_test_and_set(&(var), (val))
29
+ # define ATOMIC_INC(var) __sync_fetch_and_add(&(var), 1)
30
+ # define ATOMIC_DEC(var) __sync_fetch_and_sub(&(var), 1)
31
+ # define ATOMIC_OR(var, val) __sync_or_and_fetch(&(var), (val))
32
+ # define ATOMIC_EXCHANGE(var, val) __sync_lock_test_and_set(&(var), (val))
33
+ # define ATOMIC_CAS(var, oldval, newval) __sync_val_compare_and_swap(&(var), (oldval), (newval))
34
+
35
+ # define ATOMIC_SIZE_ADD(var, val) __sync_fetch_and_add(&(var), (val))
36
+ # define ATOMIC_SIZE_SUB(var, val) __sync_fetch_and_sub(&(var), (val))
37
+
38
+ # define ATOMIC_PTR_EXCHANGE(var, val) __sync_lock_test_and_set(&(var), (val))
39
+
40
+ #elif defined _WIN32
41
+ #if defined _MSC_VER && _MSC_VER > 1200
42
+ #pragma intrinsic(_InterlockedOr)
43
+ #endif
44
+ typedef LONG rb_atomic_t;
45
+
46
+ # define ATOMIC_SET(var, val) InterlockedExchange(&(var), (val))
47
+ # define ATOMIC_INC(var) InterlockedIncrement(&(var))
48
+ # define ATOMIC_DEC(var) InterlockedDecrement(&(var))
49
+ #if defined __GNUC__
50
+ # define ATOMIC_OR(var, val) __asm__("lock\n\t" "orl\t%1, %0" : "=m"(var) : "Ir"(val))
51
+ #elif defined _MSC_VER && _MSC_VER <= 1200
52
+ # define ATOMIC_OR(var, val) rb_w32_atomic_or(&(var), (val))
53
+ static inline void
54
+ rb_w32_atomic_or(volatile rb_atomic_t *var, rb_atomic_t val)
55
+ {
56
+ #ifdef _M_IX86
57
+ __asm mov eax, var;
58
+ __asm mov ecx, val;
59
+ __asm lock or [eax], ecx;
60
+ #else
61
+ #error unsupported architecture
62
+ #endif
63
+ }
64
+ #else
65
+ # define ATOMIC_OR(var, val) _InterlockedOr(&(var), (val))
66
+ #endif
67
+ # define ATOMIC_EXCHANGE(var, val) InterlockedExchange(&(var), (val))
68
+ # define ATOMIC_CAS(var, oldval, newval) InterlockedCompareExchange(&(var), (newval), (oldval))
69
+ # if defined _MSC_VER && _MSC_VER <= 1200
70
+ static inline rb_atomic_t
71
+ rb_w32_atomic_cas(volatile rb_atomic_t *var, rb_atomic_t oldval, rb_atomic_t newval)
72
+ {
73
+ return (rb_atomic_t)InterlockedCompareExchange((PVOID *)var, (PVOID)newval, (PVOID)oldval);
74
+ }
75
+ # undef ATOMIC_CAS
76
+ # define ATOMIC_CAS(var, oldval, newval) rb_w32_atomic_cas(&(var), (oldval), (newval))
77
+ # endif
78
+ # ifdef _M_AMD64
79
+ # define ATOMIC_SIZE_ADD(var, val) InterlockedExchangeAdd64((LONG_LONG *)&(var), (val))
80
+ # define ATOMIC_SIZE_SUB(var, val) InterlockedExchangeAdd64((LONG_LONG *)&(var), -(LONG)(val))
81
+ # define ATOMIC_SIZE_INC(var) InterlockedIncrement64(&(var))
82
+ # define ATOMIC_SIZE_DEC(var) InterlockedDecrement64(&(var))
83
+ # define ATOMIC_SIZE_EXCHANGE(var, val) InterlockedExchange64(&(var), (val))
84
+ # define ATOMIC_SIZE_CAS(var, oldval, val) InterlockedCompareExchange64(&(var), (oldval), (val))
85
+ # else
86
+ # define ATOMIC_SIZE_ADD(var, val) InterlockedExchangeAdd((LONG *)&(var), (val))
87
+ # define ATOMIC_SIZE_SUB(var, val) InterlockedExchangeAdd((LONG *)&(var), -(LONG)(val))
88
+ # define ATOMIC_SIZE_INC(var) InterlockedIncrement((LONG *)&(var))
89
+ # define ATOMIC_SIZE_DEC(var) InterlockedDecrement((LONG *)&(var))
90
+ # define ATOMIC_SIZE_EXCHANGE(var, val) InterlockedExchange((LONG *)&(var), (val))
91
+ # endif
92
+
93
+ #elif defined(__sun) && defined(HAVE_ATOMIC_H)
94
+ #include <atomic.h>
95
+ typedef unsigned int rb_atomic_t;
96
+
97
+ # define ATOMIC_SET(var, val) (void)atomic_swap_uint(&(var), (val))
98
+ # define ATOMIC_INC(var) atomic_inc_uint(&(var))
99
+ # define ATOMIC_DEC(var) atomic_dec_uint(&(var))
100
+ # define ATOMIC_OR(var, val) atomic_or_uint(&(var), (val))
101
+ # define ATOMIC_EXCHANGE(var, val) atomic_swap_uint(&(var), (val))
102
+ # define ATOMIC_CAS(var, oldval, newval) atomic_cas_uint(&(var), (oldval), (newval))
103
+
104
+ # if SIZEOF_SIZE_T == SIZEOF_LONG
105
+ # define ATOMIC_SIZE_ADD(var, val) atomic_add_long(&(var), (val))
106
+ # define ATOMIC_SIZE_SUB(var, val) atomic_add_long(&(var), -(val))
107
+ # define ATOMIC_SIZE_INC(var) atomic_inc_ulong(&(var))
108
+ # define ATOMIC_SIZE_DEC(var) atomic_dec_ulong(&(var))
109
+ # define ATOMIC_SIZE_EXCHANGE(var, val) atomic_swap_ulong(&(var), (val))
110
+ # define ATOMIC_SIZE_CAS(var, oldval, val) atomic_cas_ulong(&(var), (oldval), (val))
111
+ # else
112
+ # define ATOMIC_SIZE_ADD(var, val) atomic_add_int(&(var), (val))
113
+ # define ATOMIC_SIZE_SUB(var, val) atomic_add_int(&(var), -(val))
114
+ # define ATOMIC_SIZE_INC(var) atomic_inc_uint(&(var))
115
+ # define ATOMIC_SIZE_DEC(var) atomic_dec_uint(&(var))
116
+ # define ATOMIC_SIZE_EXCHANGE(var, val) atomic_swap_uint(&(var), (val))
117
+ # endif
118
+
119
+ #else
120
+ typedef int rb_atomic_t;
121
+ #define NEED_RUBY_ATOMIC_OPS
122
+ extern rb_atomic_t ruby_atomic_exchange(rb_atomic_t *ptr, rb_atomic_t val);
123
+ extern rb_atomic_t ruby_atomic_compare_and_swap(rb_atomic_t *ptr,
124
+ rb_atomic_t cmp,
125
+ rb_atomic_t newval);
126
+
127
+ # define ATOMIC_SET(var, val) (void)((var) = (val))
128
+ # define ATOMIC_INC(var) ((var)++)
129
+ # define ATOMIC_DEC(var) ((var)--)
130
+ # define ATOMIC_OR(var, val) ((var) |= (val))
131
+ # define ATOMIC_EXCHANGE(var, val) ruby_atomic_exchange(&(var), (val))
132
+ # define ATOMIC_CAS(var, oldval, newval) ruby_atomic_compare_and_swap(&(var), (oldval), (newval))
133
+
134
+ # define ATOMIC_SIZE_ADD(var, val) (void)((var) += (val))
135
+ # define ATOMIC_SIZE_SUB(var, val) (void)((var) -= (val))
136
+ # define ATOMIC_SIZE_EXCHANGE(var, val) atomic_size_exchange(&(var), (val))
137
+ static inline size_t
138
+ atomic_size_exchange(size_t *ptr, size_t val)
139
+ {
140
+ size_t old = *ptr;
141
+ *ptr = val;
142
+ return old;
143
+ }
144
+ #endif
145
+
146
+ #ifndef ATOMIC_SIZE_INC
147
+ # define ATOMIC_SIZE_INC(var) ATOMIC_INC(var)
148
+ #endif
149
+ #ifndef ATOMIC_SIZE_DEC
150
+ # define ATOMIC_SIZE_DEC(var) ATOMIC_DEC(var)
151
+ #endif
152
+ #ifndef ATOMIC_SIZE_EXCHANGE
153
+ # define ATOMIC_SIZE_EXCHANGE(var, val) ATOMIC_EXCHANGE(var, val)
154
+ #endif
155
+ #ifndef ATOMIC_SIZE_CAS
156
+ # define ATOMIC_SIZE_CAS(var, oldval, val) ATOMIC_CAS(var, oldval, val)
157
+ #endif
158
+
159
+ #ifndef ATOMIC_PTR_EXCHANGE
160
+ # if SIZEOF_VOIDP == SIZEOF_SIZE_T
161
+ # define ATOMIC_PTR_EXCHANGE(var, val) (void *)ATOMIC_SIZE_EXCHANGE(*(size_t *)&(var), (size_t)(val))
162
+ # endif
163
+ #endif
164
+ #ifndef ATOMIC_PTR_CAS
165
+ # if SIZEOF_VOIDP == SIZEOF_SIZE_T
166
+ # define ATOMIC_PTR_CAS(var, oldval, val) (void *)ATOMIC_SIZE_CAS(*(size_t *)&(var), (size_t)(oldval), (size_t)(val))
167
+ # endif
168
+ #endif
169
+
170
+ #endif /* RUBY_ATOMIC_H */
@@ -0,0 +1,23 @@
1
+ #ifndef RUBY_THREAD_NATIVE_H
2
+ #define RUBY_THREAD_NATIVE_H
3
+
4
+ #if defined(_WIN32)
5
+ #include "thread_win32.h"
6
+ #elif defined(HAVE_PTHREAD_H)
7
+ #include "thread_pthread.h"
8
+ #else
9
+ #error "unsupported thread type"
10
+ #endif
11
+
12
+ RUBY_SYMBOL_EXPORT_BEGIN
13
+
14
+ rb_nativethread_id_t rb_nativethread_self();
15
+
16
+ void rb_nativethread_lock_initialize(rb_nativethread_lock_t *lock);
17
+ void rb_nativethread_lock_destroy(rb_nativethread_lock_t *lock);
18
+ void rb_nativethread_lock_lock(rb_nativethread_lock_t *lock);
19
+ void rb_nativethread_lock_unlock(rb_nativethread_lock_t *lock);
20
+
21
+ RUBY_SYMBOL_EXPORT_END
22
+
23
+ #endif
@@ -0,0 +1,56 @@
1
+ /**********************************************************************
2
+
3
+ thread_pthread.h -
4
+
5
+ $Author$
6
+
7
+ Copyright (C) 2004-2007 Koichi Sasada
8
+
9
+ **********************************************************************/
10
+
11
+ #ifndef RUBY_THREAD_PTHREAD_H
12
+ #define RUBY_THREAD_PTHREAD_H
13
+
14
+ #include <pthread.h>
15
+ #ifdef HAVE_PTHREAD_NP_H
16
+ #include <pthread_np.h>
17
+ #endif
18
+ typedef pthread_t rb_nativethread_id_t;
19
+ typedef pthread_mutex_t rb_nativethread_lock_t;
20
+
21
+ typedef struct rb_thread_cond_struct {
22
+ pthread_cond_t cond;
23
+ #ifdef HAVE_CLOCKID_T
24
+ clockid_t clockid;
25
+ #endif
26
+ } rb_nativethread_cond_t;
27
+
28
+ typedef struct native_thread_data_struct {
29
+ void *signal_thread_list;
30
+ rb_nativethread_cond_t sleep_cond;
31
+ } native_thread_data_t;
32
+
33
+ #include <semaphore.h>
34
+
35
+ #undef except
36
+ #undef try
37
+ #undef leave
38
+ #undef finally
39
+
40
+ typedef struct rb_global_vm_lock_struct {
41
+ /* fast path */
42
+ unsigned long acquired;
43
+ pthread_mutex_t lock;
44
+
45
+ /* slow path */
46
+ volatile unsigned long waiting;
47
+ rb_nativethread_cond_t cond;
48
+
49
+ /* yield */
50
+ rb_nativethread_cond_t switch_cond;
51
+ rb_nativethread_cond_t switch_wait_cond;
52
+ int need_yield;
53
+ int wait_yield;
54
+ } rb_global_vm_lock_t;
55
+
56
+ #endif /* RUBY_THREAD_PTHREAD_H */
@@ -0,0 +1,45 @@
1
+ /**********************************************************************
2
+
3
+ thread_win32.h -
4
+
5
+ $Author$
6
+
7
+ Copyright (C) 2004-2007 Koichi Sasada
8
+
9
+ **********************************************************************/
10
+
11
+ /* interface */
12
+ #ifndef RUBY_THREAD_WIN32_H
13
+ #define RUBY_THREAD_WIN32_H
14
+
15
+ #include <windows.h>
16
+
17
+ # ifdef __CYGWIN__
18
+ # undef _WIN32
19
+ # endif
20
+
21
+ WINBASEAPI BOOL WINAPI
22
+ TryEnterCriticalSection(IN OUT LPCRITICAL_SECTION lpCriticalSection);
23
+
24
+ typedef HANDLE rb_nativethread_id_t;
25
+
26
+ typedef union rb_thread_lock_union {
27
+ HANDLE mutex;
28
+ CRITICAL_SECTION crit;
29
+ } rb_nativethread_lock_t;
30
+
31
+ typedef struct rb_thread_cond_struct {
32
+ struct cond_event_entry *next;
33
+ struct cond_event_entry *prev;
34
+ } rb_nativethread_cond_t;
35
+
36
+ typedef struct native_thread_data_struct {
37
+ HANDLE interrupt_event;
38
+ } native_thread_data_t;
39
+
40
+ typedef struct rb_global_vm_lock_struct {
41
+ HANDLE lock;
42
+ } rb_global_vm_lock_t;
43
+
44
+ #endif /* RUBY_THREAD_WIN32_H */
45
+
@@ -0,0 +1,1055 @@
1
+ /**********************************************************************
2
+
3
+ vm_core.h -
4
+
5
+ $Author$
6
+ created at: 04/01/01 19:41:38 JST
7
+
8
+ Copyright (C) 2004-2007 Koichi Sasada
9
+
10
+ **********************************************************************/
11
+
12
+ #ifndef RUBY_VM_CORE_H
13
+ #define RUBY_VM_CORE_H
14
+
15
+ #define RUBY_VM_THREAD_MODEL 2
16
+
17
+ #include "ruby/ruby.h"
18
+ #include "ruby/st.h"
19
+
20
+ #include "node.h"
21
+ #include "vm_debug.h"
22
+ #include "vm_opts.h"
23
+ #include "id.h"
24
+ #include "method.h"
25
+ #include "ruby_atomic.h"
26
+
27
+ #include "thread_native.h"
28
+
29
+ #ifndef ENABLE_VM_OBJSPACE
30
+ #ifdef _WIN32
31
+ /*
32
+ * TODO: object space independent st_table.
33
+ * socklist needs st_table in rb_w32_sysinit(), before object space
34
+ * initialization.
35
+ * It is too early now to change st_hash_type, since it breaks binary
36
+ * compatibility.
37
+ */
38
+ #define ENABLE_VM_OBJSPACE 0
39
+ #else
40
+ #define ENABLE_VM_OBJSPACE 1
41
+ #endif
42
+ #endif
43
+
44
+ #include <setjmp.h>
45
+ #include <signal.h>
46
+
47
+ #ifndef NSIG
48
+ # define NSIG (_SIGMAX + 1) /* For QNX */
49
+ #endif
50
+
51
+ #define RUBY_NSIG NSIG
52
+
53
+ #ifdef HAVE_STDARG_PROTOTYPES
54
+ #include <stdarg.h>
55
+ #define va_init_list(a,b) va_start((a),(b))
56
+ #else
57
+ #include <varargs.h>
58
+ #define va_init_list(a,b) va_start((a))
59
+ #endif
60
+
61
+ #if defined(SIGSEGV) && defined(HAVE_SIGALTSTACK) && defined(SA_SIGINFO) && !defined(__NetBSD__)
62
+ #define USE_SIGALTSTACK
63
+ #endif
64
+
65
+ /*****************/
66
+ /* configuration */
67
+ /*****************/
68
+
69
+ /* gcc ver. check */
70
+ #if defined(__GNUC__) && __GNUC__ >= 2
71
+
72
+ #if OPT_TOKEN_THREADED_CODE
73
+ #if OPT_DIRECT_THREADED_CODE
74
+ #undef OPT_DIRECT_THREADED_CODE
75
+ #endif
76
+ #endif
77
+
78
+ #else /* defined(__GNUC__) && __GNUC__ >= 2 */
79
+
80
+ /* disable threaded code options */
81
+ #if OPT_DIRECT_THREADED_CODE
82
+ #undef OPT_DIRECT_THREADED_CODE
83
+ #endif
84
+ #if OPT_TOKEN_THREADED_CODE
85
+ #undef OPT_TOKEN_THREADED_CODE
86
+ #endif
87
+ #endif
88
+
89
+ #ifdef __native_client__
90
+ #undef OPT_DIRECT_THREADED_CODE
91
+ #endif
92
+
93
+ /* call threaded code */
94
+ #if OPT_CALL_THREADED_CODE
95
+ #if OPT_DIRECT_THREADED_CODE
96
+ #undef OPT_DIRECT_THREADED_CODE
97
+ #endif /* OPT_DIRECT_THREADED_CODE */
98
+ #if OPT_STACK_CACHING
99
+ #undef OPT_STACK_CACHING
100
+ #endif /* OPT_STACK_CACHING */
101
+ #endif /* OPT_CALL_THREADED_CODE */
102
+
103
+ /* likely */
104
+ #if __GNUC__ >= 3
105
+ #define LIKELY(x) (__builtin_expect((x), 1))
106
+ #define UNLIKELY(x) (__builtin_expect((x), 0))
107
+ #else /* __GNUC__ >= 3 */
108
+ #define LIKELY(x) (x)
109
+ #define UNLIKELY(x) (x)
110
+ #endif /* __GNUC__ >= 3 */
111
+
112
+ #ifndef __has_attribute
113
+ # define __has_attribute(x) 0
114
+ #endif
115
+
116
+ #if __has_attribute(unused)
117
+ #define UNINITIALIZED_VAR(x) x __attribute__((unused))
118
+ #elif defined(__GNUC__) && __GNUC__ >= 3
119
+ #define UNINITIALIZED_VAR(x) x = x
120
+ #else
121
+ #define UNINITIALIZED_VAR(x) x
122
+ #endif
123
+
124
+ typedef unsigned long rb_num_t;
125
+
126
+ /* iseq data type */
127
+
128
+ struct iseq_compile_data_ensure_node_stack;
129
+
130
+ typedef struct rb_compile_option_struct rb_compile_option_t;
131
+
132
+
133
+ struct iseq_inline_cache_entry {
134
+ rb_serial_t ic_serial;
135
+ union {
136
+ size_t index;
137
+ VALUE value;
138
+ } ic_value;
139
+ };
140
+
141
+ union iseq_inline_storage_entry {
142
+ struct {
143
+ struct rb_thread_struct *running_thread;
144
+ VALUE value;
145
+ VALUE done;
146
+ } once;
147
+ struct iseq_inline_cache_entry cache;
148
+ };
149
+
150
+ /* to avoid warning */
151
+ struct rb_thread_struct;
152
+ struct rb_control_frame_struct;
153
+
154
+ /* rb_call_info_t contains calling information including inline cache */
155
+ typedef struct rb_call_info_struct {
156
+ /* fixed at compile time */
157
+ ID mid;
158
+ VALUE flag;
159
+ int orig_argc;
160
+ rb_iseq_t *blockiseq;
161
+
162
+ /* inline cache: keys */
163
+ rb_serial_t method_state;
164
+ rb_serial_t class_serial;
165
+ VALUE klass;
166
+
167
+ /* inline cache: values */
168
+ const rb_method_entry_t *me;
169
+ VALUE defined_class;
170
+
171
+ /* temporary values for method calling */
172
+ int argc;
173
+ struct rb_block_struct *blockptr;
174
+ VALUE recv;
175
+ union {
176
+ int opt_pc; /* used by iseq */
177
+ long index; /* used by ivar */
178
+ int missing_reason; /* used by method_missing */
179
+ int inc_sp; /* used by cfunc */
180
+ } aux;
181
+
182
+ VALUE (*call)(struct rb_thread_struct *th, struct rb_control_frame_struct *cfp, struct rb_call_info_struct *ci);
183
+ } rb_call_info_t;
184
+
185
+ #if 1
186
+ #define GetCoreDataFromValue(obj, type, ptr) do { \
187
+ (ptr) = (type*)DATA_PTR(obj); \
188
+ } while (0)
189
+ #else
190
+ #define GetCoreDataFromValue(obj, type, ptr) Data_Get_Struct((obj), type, (ptr))
191
+ #endif
192
+
193
+ #define GetISeqPtr(obj, ptr) \
194
+ GetCoreDataFromValue((obj), rb_iseq_t, (ptr))
195
+
196
+ typedef struct rb_iseq_location_struct {
197
+ const VALUE path;
198
+ const VALUE absolute_path;
199
+ const VALUE base_label;
200
+ const VALUE label;
201
+ size_t first_lineno;
202
+ } rb_iseq_location_t;
203
+
204
+ struct rb_iseq_struct;
205
+
206
+ struct rb_iseq_struct {
207
+ /***************/
208
+ /* static data */
209
+ /***************/
210
+
211
+ enum iseq_type {
212
+ ISEQ_TYPE_TOP,
213
+ ISEQ_TYPE_METHOD,
214
+ ISEQ_TYPE_BLOCK,
215
+ ISEQ_TYPE_CLASS,
216
+ ISEQ_TYPE_RESCUE,
217
+ ISEQ_TYPE_ENSURE,
218
+ ISEQ_TYPE_EVAL,
219
+ ISEQ_TYPE_MAIN,
220
+ ISEQ_TYPE_DEFINED_GUARD
221
+ } type; /* instruction sequence type */
222
+
223
+ rb_iseq_location_t location;
224
+
225
+ VALUE *iseq; /* iseq (insn number and operands) */
226
+ VALUE *iseq_encoded; /* encoded iseq */
227
+ unsigned long iseq_size;
228
+ const VALUE mark_ary; /* Array: includes operands which should be GC marked */
229
+ const VALUE coverage; /* coverage array */
230
+
231
+ /* insn info, must be freed */
232
+ struct iseq_line_info_entry *line_info_table;
233
+ size_t line_info_size;
234
+
235
+ ID *local_table; /* must free */
236
+ int local_table_size;
237
+
238
+ /* sizeof(vars) + 1 */
239
+ int local_size;
240
+
241
+ union iseq_inline_storage_entry *is_entries;
242
+ int is_size;
243
+
244
+ rb_call_info_t *callinfo_entries;
245
+ int callinfo_size;
246
+
247
+ /**
248
+ * argument information
249
+ *
250
+ * def m(a1, a2, ..., aM, # mandatory
251
+ * b1=(...), b2=(...), ..., bN=(...), # optional
252
+ * *c, # rest
253
+ * d1, d2, ..., dO, # post
254
+ * e1:(...), e2:(...), ..., eK:(...), # keyword
255
+ * **f, # keyword rest
256
+ * &g) # block
257
+ * =>
258
+ *
259
+ * argc = M // or 0 if no mandatory arg
260
+ * arg_opts = N+1 // or 0 if no optional arg
261
+ * arg_rest = M+N // or -1 if no rest arg
262
+ * arg_opt_table = [ (arg_opts entries) ]
263
+ * arg_post_start = M+N+(*1) // or 0 if no post arguments
264
+ * arg_post_len = O // or 0 if no post arguments
265
+ * arg_keywords = K // or 0 if no keyword arg
266
+ * arg_block = M+N+(*1)+O+K // or -1 if no block arg
267
+ * arg_keyword = M+N+(*1)+O+K+(&1) // or -1 if no keyword arg/rest
268
+ * arg_simple = 0 if not simple arguments.
269
+ * = 1 if no opt, rest, post, block.
270
+ * = 2 if ambiguous block parameter ({|a|}).
271
+ * arg_size = M+N+O+(*1)+K+(&1)+(**1) argument size.
272
+ */
273
+
274
+ int argc;
275
+ int arg_simple;
276
+ int arg_rest;
277
+ int arg_block;
278
+ int arg_opts;
279
+ int arg_post_len;
280
+ int arg_post_start;
281
+ int arg_size;
282
+ VALUE *arg_opt_table;
283
+ int arg_keyword;
284
+ int arg_keyword_check; /* if this is true, raise an ArgumentError when unknown keyword argument is passed */
285
+ int arg_keywords;
286
+ int arg_keyword_required;
287
+ ID *arg_keyword_table;
288
+
289
+ size_t stack_max; /* for stack overflow check */
290
+
291
+ /* catch table */
292
+ struct iseq_catch_table_entry *catch_table;
293
+ int catch_table_size;
294
+
295
+ /* for child iseq */
296
+ struct rb_iseq_struct *parent_iseq;
297
+ struct rb_iseq_struct *local_iseq;
298
+
299
+ /****************/
300
+ /* dynamic data */
301
+ /****************/
302
+
303
+ VALUE self;
304
+ const VALUE orig; /* non-NULL if its data have origin */
305
+
306
+ /* block inlining */
307
+ /*
308
+ * NODE *node;
309
+ * void *special_block_builder;
310
+ * void *cached_special_block_builder;
311
+ * VALUE cached_special_block;
312
+ */
313
+
314
+ /* klass/module nest information stack (cref) */
315
+ NODE * const cref_stack;
316
+ const VALUE klass;
317
+
318
+ /* misc */
319
+ ID defined_method_id; /* for define_method */
320
+ rb_num_t flip_cnt;
321
+
322
+ /* used at compile time */
323
+ struct iseq_compile_data *compile_data;
324
+ };
325
+
326
+ enum ruby_special_exceptions {
327
+ ruby_error_reenter,
328
+ ruby_error_nomemory,
329
+ ruby_error_sysstack,
330
+ ruby_error_closed_stream,
331
+ ruby_special_error_count
332
+ };
333
+
334
+ #define GetVMPtr(obj, ptr) \
335
+ GetCoreDataFromValue((obj), rb_vm_t, (ptr))
336
+
337
+ #if defined(ENABLE_VM_OBJSPACE) && ENABLE_VM_OBJSPACE
338
+ struct rb_objspace;
339
+ void rb_objspace_free(struct rb_objspace *);
340
+ #endif
341
+
342
+ typedef struct rb_hook_list_struct {
343
+ struct rb_event_hook_struct *hooks;
344
+ rb_event_flag_t events;
345
+ int need_clean;
346
+ } rb_hook_list_t;
347
+
348
+ typedef struct rb_vm_struct {
349
+ VALUE self;
350
+
351
+ rb_global_vm_lock_t gvl;
352
+ rb_nativethread_lock_t thread_destruct_lock;
353
+
354
+ struct rb_thread_struct *main_thread;
355
+ struct rb_thread_struct *running_thread;
356
+
357
+ st_table *living_threads;
358
+ VALUE thgroup_default;
359
+
360
+ int running;
361
+ int thread_abort_on_exception;
362
+ int trace_running;
363
+ volatile int sleeper;
364
+
365
+ /* object management */
366
+ VALUE mark_object_ary;
367
+
368
+ VALUE special_exceptions[ruby_special_error_count];
369
+
370
+ /* load */
371
+ VALUE top_self;
372
+ VALUE load_path;
373
+ VALUE load_path_snapshot;
374
+ VALUE load_path_check_cache;
375
+ VALUE expanded_load_path;
376
+ VALUE loaded_features;
377
+ VALUE loaded_features_snapshot;
378
+ struct st_table *loaded_features_index;
379
+ struct st_table *loading_table;
380
+
381
+ /* signal */
382
+ struct {
383
+ VALUE cmd;
384
+ int safe;
385
+ } trap_list[RUBY_NSIG];
386
+
387
+ /* hook */
388
+ rb_hook_list_t event_hooks;
389
+
390
+ /* relation table of ensure - rollback for callcc */
391
+ struct st_table *ensure_rollback_table;
392
+
393
+ /* postponed_job */
394
+ struct rb_postponed_job_struct *postponed_job_buffer;
395
+ int postponed_job_index;
396
+
397
+ int src_encoding_index;
398
+
399
+ VALUE verbose, debug, orig_progname, progname;
400
+ VALUE coverages;
401
+
402
+ struct unlinked_method_entry_list_entry *unlinked_method_entry_list;
403
+
404
+ VALUE defined_module_hash;
405
+
406
+ #if defined(ENABLE_VM_OBJSPACE) && ENABLE_VM_OBJSPACE
407
+ struct rb_objspace *objspace;
408
+ #endif
409
+
410
+ /*
411
+ * @shyouhei notes that this is not for storing normal Ruby
412
+ * objects so do *NOT* mark this when you GC.
413
+ */
414
+ struct RArray at_exit;
415
+
416
+ VALUE *defined_strings;
417
+
418
+ /* params */
419
+ struct { /* size in byte */
420
+ size_t thread_vm_stack_size;
421
+ size_t thread_machine_stack_size;
422
+ size_t fiber_vm_stack_size;
423
+ size_t fiber_machine_stack_size;
424
+ } default_params;
425
+ } rb_vm_t;
426
+
427
+ /* default values */
428
+
429
+ #define RUBY_VM_SIZE_ALIGN 4096
430
+
431
+ #define RUBY_VM_THREAD_VM_STACK_SIZE ( 128 * 1024 * sizeof(VALUE)) /* 512 KB or 1024 KB */
432
+ #define RUBY_VM_THREAD_VM_STACK_SIZE_MIN ( 2 * 1024 * sizeof(VALUE)) /* 8 KB or 16 KB */
433
+ #define RUBY_VM_THREAD_MACHINE_STACK_SIZE ( 128 * 1024 * sizeof(VALUE)) /* 512 KB or 1024 KB */
434
+ #define RUBY_VM_THREAD_MACHINE_STACK_SIZE_MIN ( 16 * 1024 * sizeof(VALUE)) /* 64 KB or 128 KB */
435
+
436
+ #define RUBY_VM_FIBER_VM_STACK_SIZE ( 16 * 1024 * sizeof(VALUE)) /* 64 KB or 128 KB */
437
+ #define RUBY_VM_FIBER_VM_STACK_SIZE_MIN ( 2 * 1024 * sizeof(VALUE)) /* 8 KB or 16 KB */
438
+ #define RUBY_VM_FIBER_MACHINE_STACK_SIZE ( 64 * 1024 * sizeof(VALUE)) /* 256 KB or 512 KB */
439
+ #define RUBY_VM_FIBER_MACHINE_STACK_SIZE_MIN ( 16 * 1024 * sizeof(VALUE)) /* 64 KB or 128 KB */
440
+
441
+ #ifndef VM_DEBUG_BP_CHECK
442
+ #define VM_DEBUG_BP_CHECK 0
443
+ #endif
444
+
445
+ typedef struct rb_control_frame_struct {
446
+ VALUE *pc; /* cfp[0] */
447
+ VALUE *sp; /* cfp[1] */
448
+ rb_iseq_t *iseq; /* cfp[2] */
449
+ VALUE flag; /* cfp[3] */
450
+ VALUE self; /* cfp[4] / block[0] */
451
+ VALUE klass; /* cfp[5] / block[1] */
452
+ VALUE *ep; /* cfp[6] / block[2] */
453
+ rb_iseq_t *block_iseq; /* cfp[7] / block[3] */
454
+ VALUE proc; /* cfp[8] / block[4] */
455
+ const rb_method_entry_t *me;/* cfp[9] */
456
+
457
+ #if VM_DEBUG_BP_CHECK
458
+ VALUE *bp_check; /* cfp[10] */
459
+ #endif
460
+ } rb_control_frame_t;
461
+
462
+ typedef struct rb_block_struct {
463
+ VALUE self; /* share with method frame if it's only block */
464
+ VALUE klass; /* share with method frame if it's only block */
465
+ VALUE *ep; /* share with method frame if it's only block */
466
+ rb_iseq_t *iseq;
467
+ VALUE proc;
468
+ } rb_block_t;
469
+
470
+ extern const rb_data_type_t ruby_threadptr_data_type;
471
+
472
+ #define GetThreadPtr(obj, ptr) \
473
+ TypedData_Get_Struct((obj), rb_thread_t, &ruby_threadptr_data_type, (ptr))
474
+
475
+ enum rb_thread_status {
476
+ THREAD_RUNNABLE,
477
+ THREAD_STOPPED,
478
+ THREAD_STOPPED_FOREVER,
479
+ THREAD_KILLED
480
+ };
481
+
482
+ typedef RUBY_JMP_BUF rb_jmpbuf_t;
483
+
484
+ /*
485
+ the members which are written in TH_PUSH_TAG() should be placed at
486
+ the beginning and the end, so that entire region is accessible.
487
+ */
488
+ struct rb_vm_tag {
489
+ VALUE tag;
490
+ VALUE retval;
491
+ rb_jmpbuf_t buf;
492
+ struct rb_vm_tag *prev;
493
+ };
494
+
495
+ struct rb_vm_protect_tag {
496
+ struct rb_vm_protect_tag *prev;
497
+ };
498
+
499
+ struct rb_unblock_callback {
500
+ rb_unblock_function_t *func;
501
+ void *arg;
502
+ };
503
+
504
+ struct rb_mutex_struct;
505
+
506
+ struct rb_thread_struct;
507
+ typedef struct rb_thread_list_struct{
508
+ struct rb_thread_list_struct *next;
509
+ struct rb_thread_struct *th;
510
+ } rb_thread_list_t;
511
+
512
+
513
+ typedef struct rb_ensure_entry {
514
+ VALUE marker;
515
+ VALUE (*e_proc)(ANYARGS);
516
+ VALUE data2;
517
+ } rb_ensure_entry_t;
518
+
519
+ typedef struct rb_ensure_list {
520
+ struct rb_ensure_list *next;
521
+ struct rb_ensure_entry entry;
522
+ } rb_ensure_list_t;
523
+
524
+ typedef struct rb_thread_struct {
525
+ VALUE self;
526
+ rb_vm_t *vm;
527
+
528
+ /* execution information */
529
+ VALUE *stack; /* must free, must mark */
530
+ size_t stack_size; /* size in word (byte size / sizeof(VALUE)) */
531
+ rb_control_frame_t *cfp;
532
+ int safe_level;
533
+ int raised_flag;
534
+ VALUE last_status; /* $? */
535
+
536
+ /* passing state */
537
+ int state;
538
+
539
+ int waiting_fd;
540
+
541
+ /* for rb_iterate */
542
+ const rb_block_t *passed_block;
543
+
544
+ /* for bmethod */
545
+ const rb_method_entry_t *passed_bmethod_me;
546
+
547
+ /* for cfunc */
548
+ rb_call_info_t *passed_ci;
549
+
550
+ /* for load(true) */
551
+ VALUE top_self;
552
+ VALUE top_wrapper;
553
+
554
+ /* eval env */
555
+ rb_block_t *base_block;
556
+
557
+ VALUE *root_lep;
558
+ VALUE root_svar;
559
+
560
+ /* thread control */
561
+ rb_nativethread_id_t thread_id;
562
+ enum rb_thread_status status;
563
+ int to_kill;
564
+ int priority;
565
+
566
+ native_thread_data_t native_thread_data;
567
+ void *blocking_region_buffer;
568
+
569
+ VALUE thgroup;
570
+ VALUE value;
571
+
572
+ /* temporary place of errinfo */
573
+ VALUE errinfo;
574
+
575
+ /* temporary place of retval on OPT_CALL_THREADED_CODE */
576
+ #if OPT_CALL_THREADED_CODE
577
+ VALUE retval;
578
+ #endif
579
+
580
+ /* async errinfo queue */
581
+ VALUE pending_interrupt_queue;
582
+ int pending_interrupt_queue_checked;
583
+ VALUE pending_interrupt_mask_stack;
584
+
585
+ rb_atomic_t interrupt_flag;
586
+ unsigned long interrupt_mask;
587
+ rb_nativethread_lock_t interrupt_lock;
588
+ rb_nativethread_cond_t interrupt_cond;
589
+ struct rb_unblock_callback unblock;
590
+ VALUE locking_mutex;
591
+ struct rb_mutex_struct *keeping_mutexes;
592
+
593
+ struct rb_vm_tag *tag;
594
+ struct rb_vm_protect_tag *protect_tag;
595
+
596
+ /*! Thread-local state of evaluation context.
597
+ *
598
+ * If negative, this thread is evaluating the main program.
599
+ * If positive, this thread is evaluating a program under Kernel::eval
600
+ * family.
601
+ */
602
+ int parse_in_eval;
603
+
604
+ /*! Thread-local state of compiling context.
605
+ *
606
+ * If non-zero, the parser does not automatically print error messages to
607
+ * stderr. */
608
+ int mild_compile_error;
609
+
610
+ /* storage */
611
+ st_table *local_storage;
612
+
613
+ rb_thread_list_t *join_list;
614
+
615
+ VALUE first_proc;
616
+ VALUE first_args;
617
+ VALUE (*first_func)(ANYARGS);
618
+
619
+ /* for GC */
620
+ struct {
621
+ VALUE *stack_start;
622
+ VALUE *stack_end;
623
+ size_t stack_maxsize;
624
+ #ifdef __ia64
625
+ VALUE *register_stack_start;
626
+ VALUE *register_stack_end;
627
+ size_t register_stack_maxsize;
628
+ #endif
629
+ jmp_buf regs;
630
+ } machine;
631
+ int mark_stack_len;
632
+
633
+ /* statistics data for profiler */
634
+ VALUE stat_insn_usage;
635
+
636
+ /* tracer */
637
+ rb_hook_list_t event_hooks;
638
+ struct rb_trace_arg_struct *trace_arg; /* trace information */
639
+
640
+ /* fiber */
641
+ VALUE fiber;
642
+ VALUE root_fiber;
643
+ rb_jmpbuf_t root_jmpbuf;
644
+
645
+ /* ensure & callcc */
646
+ rb_ensure_list_t *ensure_list;
647
+
648
+ /* misc */
649
+ int method_missing_reason;
650
+ int abort_on_exception;
651
+ #ifdef USE_SIGALTSTACK
652
+ void *altstack;
653
+ #endif
654
+ unsigned long running_time_us;
655
+ } rb_thread_t;
656
+
657
+ typedef enum {
658
+ VM_DEFINECLASS_TYPE_CLASS = 0x00,
659
+ VM_DEFINECLASS_TYPE_SINGLETON_CLASS = 0x01,
660
+ VM_DEFINECLASS_TYPE_MODULE = 0x02,
661
+ /* 0x03..0x06 is reserved */
662
+ VM_DEFINECLASS_TYPE_MASK = 0x07
663
+ } rb_vm_defineclass_type_t;
664
+
665
+ #define VM_DEFINECLASS_TYPE(x) ((rb_vm_defineclass_type_t)(x) & VM_DEFINECLASS_TYPE_MASK)
666
+ #define VM_DEFINECLASS_FLAG_SCOPED 0x08
667
+ #define VM_DEFINECLASS_FLAG_HAS_SUPERCLASS 0x10
668
+ #define VM_DEFINECLASS_SCOPED_P(x) ((x) & VM_DEFINECLASS_FLAG_SCOPED)
669
+ #define VM_DEFINECLASS_HAS_SUPERCLASS_P(x) \
670
+ ((x) & VM_DEFINECLASS_FLAG_HAS_SUPERCLASS)
671
+
672
+ /* iseq.c */
673
+ RUBY_SYMBOL_EXPORT_BEGIN
674
+
675
+ /* node -> iseq */
676
+ VALUE rb_iseq_new(NODE*, VALUE, VALUE, VALUE, VALUE, enum iseq_type);
677
+ VALUE rb_iseq_new_top(NODE *node, VALUE name, VALUE path, VALUE absolute_path, VALUE parent);
678
+ VALUE rb_iseq_new_main(NODE *node, VALUE path, VALUE absolute_path);
679
+ VALUE rb_iseq_new_with_bopt(NODE*, VALUE, VALUE, VALUE, VALUE, VALUE, enum iseq_type, VALUE);
680
+ VALUE rb_iseq_new_with_opt(NODE*, VALUE, VALUE, VALUE, VALUE, VALUE, enum iseq_type, const rb_compile_option_t*);
681
+
682
+ /* src -> iseq */
683
+ VALUE rb_iseq_compile(VALUE src, VALUE file, VALUE line);
684
+ VALUE rb_iseq_compile_on_base(VALUE src, VALUE file, VALUE line, rb_block_t *base_block);
685
+ VALUE rb_iseq_compile_with_option(VALUE src, VALUE file, VALUE absolute_path, VALUE line, rb_block_t *base_block, VALUE opt);
686
+
687
+ VALUE rb_iseq_disasm(VALUE self);
688
+ int rb_iseq_disasm_insn(VALUE str, VALUE *iseqval, size_t pos, rb_iseq_t *iseq, VALUE child);
689
+ const char *ruby_node_name(int node);
690
+
691
+ RUBY_EXTERN VALUE rb_cISeq;
692
+ RUBY_EXTERN VALUE rb_cRubyVM;
693
+ RUBY_EXTERN VALUE rb_cEnv;
694
+ RUBY_EXTERN VALUE rb_mRubyVMFrozenCore;
695
+ RUBY_SYMBOL_EXPORT_END
696
+
697
+ #define GetProcPtr(obj, ptr) \
698
+ GetCoreDataFromValue((obj), rb_proc_t, (ptr))
699
+
700
+ typedef struct {
701
+ rb_block_t block;
702
+
703
+ VALUE envval; /* for GC mark */
704
+ VALUE blockprocval;
705
+ int safe_level;
706
+ int is_from_method;
707
+ int is_lambda;
708
+ } rb_proc_t;
709
+
710
+ #define GetEnvPtr(obj, ptr) \
711
+ GetCoreDataFromValue((obj), rb_env_t, (ptr))
712
+
713
+ typedef struct {
714
+ VALUE *env;
715
+ int env_size;
716
+ int local_size;
717
+ VALUE prev_envval; /* for GC mark */
718
+ rb_block_t block;
719
+ } rb_env_t;
720
+
721
+ extern const rb_data_type_t ruby_binding_data_type;
722
+
723
+ #define GetBindingPtr(obj, ptr) \
724
+ GetCoreDataFromValue((obj), rb_binding_t, (ptr))
725
+
726
+ typedef struct {
727
+ VALUE env;
728
+ VALUE path;
729
+ VALUE blockprocval; /* for GC mark */
730
+ unsigned short first_lineno;
731
+ } rb_binding_t;
732
+
733
+ /* used by compile time and send insn */
734
+
735
+ enum vm_check_match_type {
736
+ VM_CHECKMATCH_TYPE_WHEN = 1,
737
+ VM_CHECKMATCH_TYPE_CASE = 2,
738
+ VM_CHECKMATCH_TYPE_RESCUE = 3
739
+ };
740
+
741
+ #define VM_CHECKMATCH_TYPE_MASK 0x03
742
+ #define VM_CHECKMATCH_ARRAY 0x04
743
+
744
+ #define VM_CALL_ARGS_SPLAT (0x01 << 1) /* m(*args) */
745
+ #define VM_CALL_ARGS_BLOCKARG (0x01 << 2) /* m(&block) */
746
+ #define VM_CALL_FCALL (0x01 << 3) /* m(...) */
747
+ #define VM_CALL_VCALL (0x01 << 4) /* m */
748
+ #define VM_CALL_TAILCALL (0x01 << 5) /* located at tail position */
749
+ #define VM_CALL_SUPER (0x01 << 6) /* super */
750
+ #define VM_CALL_OPT_SEND (0x01 << 7) /* internal flag */
751
+ #define VM_CALL_ARGS_SKIP_SETUP (0x01 << 8) /* (flag & (SPLAT|BLOCKARG)) && blockiseq == 0 */
752
+
753
+ enum vm_special_object_type {
754
+ VM_SPECIAL_OBJECT_VMCORE = 1,
755
+ VM_SPECIAL_OBJECT_CBASE,
756
+ VM_SPECIAL_OBJECT_CONST_BASE
757
+ };
758
+
759
+ #define VM_FRAME_MAGIC_METHOD 0x11
760
+ #define VM_FRAME_MAGIC_BLOCK 0x21
761
+ #define VM_FRAME_MAGIC_CLASS 0x31
762
+ #define VM_FRAME_MAGIC_TOP 0x41
763
+ #define VM_FRAME_MAGIC_CFUNC 0x61
764
+ #define VM_FRAME_MAGIC_PROC 0x71
765
+ #define VM_FRAME_MAGIC_IFUNC 0x81
766
+ #define VM_FRAME_MAGIC_EVAL 0x91
767
+ #define VM_FRAME_MAGIC_LAMBDA 0xa1
768
+ #define VM_FRAME_MAGIC_RESCUE 0xb1
769
+ #define VM_FRAME_MAGIC_MASK_BITS 8
770
+ #define VM_FRAME_MAGIC_MASK (~(~0<<VM_FRAME_MAGIC_MASK_BITS))
771
+
772
+ #define VM_FRAME_TYPE(cfp) ((cfp)->flag & VM_FRAME_MAGIC_MASK)
773
+
774
+ /* other frame flag */
775
+ #define VM_FRAME_FLAG_PASSED 0x0100
776
+ #define VM_FRAME_FLAG_FINISH 0x0200
777
+ #define VM_FRAME_FLAG_BMETHOD 0x0400
778
+ #define VM_FRAME_TYPE_FINISH_P(cfp) (((cfp)->flag & VM_FRAME_FLAG_FINISH) != 0)
779
+ #define VM_FRAME_TYPE_BMETHOD_P(cfp) (((cfp)->flag & VM_FRAME_FLAG_BMETHOD) != 0)
780
+
781
+ #define RUBYVM_CFUNC_FRAME_P(cfp) \
782
+ (VM_FRAME_TYPE(cfp) == VM_FRAME_MAGIC_CFUNC)
783
+
784
+ /* inline cache */
785
+ typedef struct iseq_inline_cache_entry *IC;
786
+ typedef rb_call_info_t *CALL_INFO;
787
+
788
+ void rb_vm_change_state(void);
789
+
790
+ typedef VALUE CDHASH;
791
+
792
+ #ifndef FUNC_FASTCALL
793
+ #define FUNC_FASTCALL(x) x
794
+ #endif
795
+
796
+ typedef rb_control_frame_t *
797
+ (FUNC_FASTCALL(*rb_insn_func_t))(rb_thread_t *, rb_control_frame_t *);
798
+
799
+ #define GC_GUARDED_PTR(p) ((VALUE)((VALUE)(p) | 0x01))
800
+ #define GC_GUARDED_PTR_REF(p) ((void *)(((VALUE)(p)) & ~0x03))
801
+ #define GC_GUARDED_PTR_P(p) (((VALUE)(p)) & 0x01)
802
+
803
+ /*
804
+ * block frame:
805
+ * ep[ 0]: prev frame
806
+ * ep[-1]: CREF (for *_eval)
807
+ *
808
+ * method frame:
809
+ * ep[ 0]: block pointer (ptr | VM_ENVVAL_BLOCK_PTR_FLAG)
810
+ */
811
+
812
+ #define VM_ENVVAL_BLOCK_PTR_FLAG 0x02
813
+ #define VM_ENVVAL_BLOCK_PTR(v) (GC_GUARDED_PTR(v) | VM_ENVVAL_BLOCK_PTR_FLAG)
814
+ #define VM_ENVVAL_BLOCK_PTR_P(v) ((v) & VM_ENVVAL_BLOCK_PTR_FLAG)
815
+ #define VM_ENVVAL_PREV_EP_PTR(v) ((VALUE)GC_GUARDED_PTR(v))
816
+ #define VM_ENVVAL_PREV_EP_PTR_P(v) (!(VM_ENVVAL_BLOCK_PTR_P(v)))
817
+
818
+ #define VM_EP_PREV_EP(ep) ((VALUE *)GC_GUARDED_PTR_REF((ep)[0]))
819
+ #define VM_EP_BLOCK_PTR(ep) ((rb_block_t *)GC_GUARDED_PTR_REF((ep)[0]))
820
+ #define VM_EP_LEP_P(ep) VM_ENVVAL_BLOCK_PTR_P((ep)[0])
821
+
822
+ VALUE *rb_vm_ep_local_ep(VALUE *ep);
823
+ rb_block_t *rb_vm_control_frame_block_ptr(rb_control_frame_t *cfp);
824
+
825
+ #define RUBY_VM_PREVIOUS_CONTROL_FRAME(cfp) ((cfp)+1)
826
+ #define RUBY_VM_NEXT_CONTROL_FRAME(cfp) ((cfp)-1)
827
+ #define RUBY_VM_END_CONTROL_FRAME(th) \
828
+ ((rb_control_frame_t *)((th)->stack + (th)->stack_size))
829
+ #define RUBY_VM_VALID_CONTROL_FRAME_P(cfp, ecfp) \
830
+ ((void *)(ecfp) > (void *)(cfp))
831
+ #define RUBY_VM_CONTROL_FRAME_STACK_OVERFLOW_P(th, cfp) \
832
+ (!RUBY_VM_VALID_CONTROL_FRAME_P((cfp), RUBY_VM_END_CONTROL_FRAME(th)))
833
+
834
+ #define RUBY_VM_IFUNC_P(ptr) (BUILTIN_TYPE(ptr) == T_NODE)
835
+ #define RUBY_VM_NORMAL_ISEQ_P(ptr) \
836
+ ((ptr) && !RUBY_VM_IFUNC_P(ptr))
837
+
838
+ #define RUBY_VM_GET_BLOCK_PTR_IN_CFP(cfp) ((rb_block_t *)(&(cfp)->self))
839
+ #define RUBY_VM_GET_CFP_FROM_BLOCK_PTR(b) \
840
+ ((rb_control_frame_t *)((VALUE *)(b) - 4))
841
+ /* magic number `4' is depend on rb_control_frame_t layout. */
842
+
843
+ /* VM related object allocate functions */
844
+ VALUE rb_thread_alloc(VALUE klass);
845
+ VALUE rb_proc_alloc(VALUE klass);
846
+ VALUE rb_binding_alloc(VALUE klass);
847
+
848
+ /* for debug */
849
+ extern void rb_vmdebug_stack_dump_raw(rb_thread_t *, rb_control_frame_t *);
850
+ extern void rb_vmdebug_debug_print_pre(rb_thread_t *th, rb_control_frame_t *cfp, VALUE *_pc);
851
+ extern void rb_vmdebug_debug_print_post(rb_thread_t *th, rb_control_frame_t *cfp);
852
+
853
+ #define SDR() rb_vmdebug_stack_dump_raw(GET_THREAD(), GET_THREAD()->cfp)
854
+ #define SDR2(cfp) rb_vmdebug_stack_dump_raw(GET_THREAD(), (cfp))
855
+ void rb_vm_bugreport(void);
856
+
857
+ /* functions about thread/vm execution */
858
+ RUBY_SYMBOL_EXPORT_BEGIN
859
+ VALUE rb_iseq_eval(VALUE iseqval);
860
+ VALUE rb_iseq_eval_main(VALUE iseqval);
861
+ RUBY_SYMBOL_EXPORT_END
862
+ int rb_thread_method_id_and_class(rb_thread_t *th, ID *idp, VALUE *klassp);
863
+
864
+ VALUE rb_vm_invoke_proc(rb_thread_t *th, rb_proc_t *proc,
865
+ int argc, const VALUE *argv, const rb_block_t *blockptr);
866
+ VALUE rb_vm_make_proc(rb_thread_t *th, const rb_block_t *block, VALUE klass);
867
+ VALUE rb_vm_make_binding(rb_thread_t *th, const rb_control_frame_t *src_cfp);
868
+ VALUE rb_vm_make_env_object(rb_thread_t *th, rb_control_frame_t *cfp);
869
+ VALUE rb_binding_new_with_cfp(rb_thread_t *th, const rb_control_frame_t *src_cfp);
870
+ VALUE *rb_binding_add_dynavars(rb_binding_t *bind, int dyncount, const ID *dynvars);
871
+ void rb_vm_inc_const_missing_count(void);
872
+ void rb_vm_gvl_destroy(rb_vm_t *vm);
873
+ VALUE rb_vm_call(rb_thread_t *th, VALUE recv, VALUE id, int argc,
874
+ const VALUE *argv, const rb_method_entry_t *me,
875
+ VALUE defined_class);
876
+ void rb_unlink_method_entry(rb_method_entry_t *me);
877
+ void rb_gc_mark_unlinked_live_method_entries(void *pvm);
878
+
879
+ void rb_thread_start_timer_thread(void);
880
+ void rb_thread_stop_timer_thread(int);
881
+ void rb_thread_reset_timer_thread(void);
882
+ void rb_thread_wakeup_timer_thread(void);
883
+
884
+ int ruby_thread_has_gvl_p(void);
885
+ typedef int rb_backtrace_iter_func(void *, VALUE, int, VALUE);
886
+ rb_control_frame_t *rb_vm_get_ruby_level_next_cfp(rb_thread_t *th, const rb_control_frame_t *cfp);
887
+ rb_control_frame_t *rb_vm_get_binding_creatable_next_cfp(rb_thread_t *th, const rb_control_frame_t *cfp);
888
+ int rb_vm_get_sourceline(const rb_control_frame_t *);
889
+ VALUE rb_name_err_mesg_new(VALUE obj, VALUE mesg, VALUE recv, VALUE method);
890
+ void rb_vm_stack_to_heap(rb_thread_t *th);
891
+ void ruby_thread_init_stack(rb_thread_t *th);
892
+ int rb_vm_control_frame_id_and_class(const rb_control_frame_t *cfp, ID *idp, VALUE *klassp);
893
+ void rb_vm_rewind_cfp(rb_thread_t *th, rb_control_frame_t *cfp);
894
+
895
+ void rb_gc_mark_machine_stack(rb_thread_t *th);
896
+
897
+ int rb_autoloading_value(VALUE mod, ID id, VALUE* value);
898
+
899
+ #define sysstack_error GET_VM()->special_exceptions[ruby_error_sysstack]
900
+
901
+ #define RUBY_CONST_ASSERT(expr) (1/!!(expr)) /* expr must be a compile-time constant */
902
+ #define VM_STACK_OVERFLOWED_P(cfp, sp, margin) \
903
+ (!RUBY_CONST_ASSERT(sizeof(*(sp)) == sizeof(VALUE)) || \
904
+ !RUBY_CONST_ASSERT(sizeof(*(cfp)) == sizeof(rb_control_frame_t)) || \
905
+ ((rb_control_frame_t *)((sp) + (margin)) + 1) >= (cfp))
906
+ #define WHEN_VM_STACK_OVERFLOWED(cfp, sp, margin) \
907
+ if (LIKELY(!VM_STACK_OVERFLOWED_P(cfp, sp, margin))) {(void)0;} else /* overflowed */
908
+ #define CHECK_VM_STACK_OVERFLOW0(cfp, sp, margin) \
909
+ WHEN_VM_STACK_OVERFLOWED(cfp, sp, margin) vm_stackoverflow()
910
+ #define CHECK_VM_STACK_OVERFLOW(cfp, margin) \
911
+ WHEN_VM_STACK_OVERFLOWED(cfp, (cfp)->sp, margin) vm_stackoverflow()
912
+
913
+ /* for thread */
914
+
915
+ #if RUBY_VM_THREAD_MODEL == 2
916
+ extern rb_thread_t *ruby_current_thread;
917
+ extern rb_vm_t *ruby_current_vm;
918
+ extern rb_event_flag_t ruby_vm_event_flags;
919
+
920
+ #define GET_VM() ruby_current_vm
921
+
922
+ #ifndef OPT_CALL_CFUNC_WITHOUT_FRAME
923
+ #define OPT_CALL_CFUNC_WITHOUT_FRAME 0
924
+ #endif
925
+
926
+ static inline rb_thread_t *
927
+ GET_THREAD(void)
928
+ {
929
+ rb_thread_t *th = ruby_current_thread;
930
+ #if OPT_CALL_CFUNC_WITHOUT_FRAME
931
+ if (UNLIKELY(th->passed_ci != 0)) {
932
+ void vm_call_cfunc_push_frame(rb_thread_t *th);
933
+ vm_call_cfunc_push_frame(th);
934
+ }
935
+ #endif
936
+ return th;
937
+ }
938
+
939
+ #define rb_thread_set_current_raw(th) (void)(ruby_current_thread = (th))
940
+ #define rb_thread_set_current(th) do { \
941
+ if ((th)->vm->running_thread != (th)) { \
942
+ (th)->running_time_us = 0; \
943
+ } \
944
+ rb_thread_set_current_raw(th); \
945
+ (th)->vm->running_thread = (th); \
946
+ } while (0)
947
+
948
+ #else
949
+ #error "unsupported thread model"
950
+ #endif
951
+
952
+ enum {
953
+ TIMER_INTERRUPT_MASK = 0x01,
954
+ PENDING_INTERRUPT_MASK = 0x02,
955
+ POSTPONED_JOB_INTERRUPT_MASK = 0x04,
956
+ TRAP_INTERRUPT_MASK = 0x08
957
+ };
958
+
959
+ #define RUBY_VM_SET_TIMER_INTERRUPT(th) ATOMIC_OR((th)->interrupt_flag, TIMER_INTERRUPT_MASK)
960
+ #define RUBY_VM_SET_INTERRUPT(th) ATOMIC_OR((th)->interrupt_flag, PENDING_INTERRUPT_MASK)
961
+ #define RUBY_VM_SET_POSTPONED_JOB_INTERRUPT(th) ATOMIC_OR((th)->interrupt_flag, POSTPONED_JOB_INTERRUPT_MASK)
962
+ #define RUBY_VM_SET_TRAP_INTERRUPT(th) ATOMIC_OR((th)->interrupt_flag, TRAP_INTERRUPT_MASK)
963
+ #define RUBY_VM_INTERRUPTED(th) ((th)->interrupt_flag & ~(th)->interrupt_mask & (PENDING_INTERRUPT_MASK|TRAP_INTERRUPT_MASK))
964
+ #define RUBY_VM_INTERRUPTED_ANY(th) ((th)->interrupt_flag & ~(th)->interrupt_mask)
965
+
966
+ int rb_signal_buff_size(void);
967
+ void rb_signal_exec(rb_thread_t *th, int sig);
968
+ void rb_threadptr_check_signal(rb_thread_t *mth);
969
+ void rb_threadptr_signal_raise(rb_thread_t *th, int sig);
970
+ void rb_threadptr_signal_exit(rb_thread_t *th);
971
+ void rb_threadptr_execute_interrupts(rb_thread_t *, int);
972
+ void rb_threadptr_interrupt(rb_thread_t *th);
973
+ void rb_threadptr_unlock_all_locking_mutexes(rb_thread_t *th);
974
+ void rb_threadptr_pending_interrupt_clear(rb_thread_t *th);
975
+ void rb_threadptr_pending_interrupt_enque(rb_thread_t *th, VALUE v);
976
+ int rb_threadptr_pending_interrupt_active_p(rb_thread_t *th);
977
+
978
+ #define RUBY_VM_CHECK_INTS_BLOCKING(th) do { \
979
+ if (UNLIKELY(!rb_threadptr_pending_interrupt_empty_p(th))) { \
980
+ th->pending_interrupt_queue_checked = 0; \
981
+ RUBY_VM_SET_INTERRUPT(th); \
982
+ rb_threadptr_execute_interrupts(th, 1); \
983
+ } \
984
+ else if (UNLIKELY(RUBY_VM_INTERRUPTED_ANY(th))) { \
985
+ rb_threadptr_execute_interrupts(th, 1); \
986
+ } \
987
+ } while (0)
988
+
989
+ #define RUBY_VM_CHECK_INTS(th) do { \
990
+ if (UNLIKELY(RUBY_VM_INTERRUPTED_ANY(th))) { \
991
+ rb_threadptr_execute_interrupts(th, 0); \
992
+ } \
993
+ } while (0)
994
+
995
+ /* tracer */
996
+ struct rb_trace_arg_struct {
997
+ rb_event_flag_t event;
998
+ rb_thread_t *th;
999
+ rb_control_frame_t *cfp;
1000
+ VALUE self;
1001
+ ID id;
1002
+ VALUE klass;
1003
+ VALUE data;
1004
+
1005
+ int klass_solved;
1006
+
1007
+ /* calc from cfp */
1008
+ int lineno;
1009
+ VALUE path;
1010
+ };
1011
+
1012
+ void rb_threadptr_exec_event_hooks(struct rb_trace_arg_struct *trace_arg);
1013
+ void rb_threadptr_exec_event_hooks_and_pop_frame(struct rb_trace_arg_struct *trace_arg);
1014
+
1015
+ #define EXEC_EVENT_HOOK_ORIG(th_, flag_, self_, id_, klass_, data_, pop_p_) do { \
1016
+ if (UNLIKELY(ruby_vm_event_flags & (flag_))) { \
1017
+ if (((th)->event_hooks.events | (th)->vm->event_hooks.events) & (flag_)) { \
1018
+ struct rb_trace_arg_struct trace_arg; \
1019
+ trace_arg.event = (flag_); \
1020
+ trace_arg.th = (th_); \
1021
+ trace_arg.cfp = (trace_arg.th)->cfp; \
1022
+ trace_arg.self = (self_); \
1023
+ trace_arg.id = (id_); \
1024
+ trace_arg.klass = (klass_); \
1025
+ trace_arg.data = (data_); \
1026
+ trace_arg.path = Qundef; \
1027
+ trace_arg.klass_solved = 0; \
1028
+ if (pop_p_) rb_threadptr_exec_event_hooks_and_pop_frame(&trace_arg); \
1029
+ else rb_threadptr_exec_event_hooks(&trace_arg); \
1030
+ } \
1031
+ } \
1032
+ } while (0)
1033
+
1034
+ #define EXEC_EVENT_HOOK(th_, flag_, self_, id_, klass_, data_) \
1035
+ EXEC_EVENT_HOOK_ORIG(th_, flag_, self_, id_, klass_, data_, 0)
1036
+
1037
+ #define EXEC_EVENT_HOOK_AND_POP_FRAME(th_, flag_, self_, id_, klass_, data_) \
1038
+ EXEC_EVENT_HOOK_ORIG(th_, flag_, self_, id_, klass_, data_, 1)
1039
+
1040
+ VALUE rb_threadptr_reset_recursive_data(rb_thread_t *th);
1041
+ void rb_threadptr_restore_recursive_data(rb_thread_t *th, VALUE old);
1042
+
1043
+ RUBY_SYMBOL_EXPORT_BEGIN
1044
+
1045
+ int rb_thread_check_trap_pending(void);
1046
+
1047
+ extern VALUE rb_get_coverages(void);
1048
+ extern void rb_set_coverages(VALUE);
1049
+ extern void rb_reset_coverages(void);
1050
+
1051
+ void rb_postponed_job_flush(rb_vm_t *vm);
1052
+
1053
+ RUBY_SYMBOL_EXPORT_END
1054
+
1055
+ #endif /* RUBY_VM_CORE_H */