libv8-sgonyea 3.3.10

Sign up to get free protection for your applications and to get access to all the features.
Files changed (500) hide show
  1. data/.gitignore +8 -0
  2. data/.gitmodules +3 -0
  3. data/Gemfile +4 -0
  4. data/README.md +76 -0
  5. data/Rakefile +113 -0
  6. data/ext/libv8/extconf.rb +28 -0
  7. data/lib/libv8.rb +15 -0
  8. data/lib/libv8/Makefile +30 -0
  9. data/lib/libv8/detect_cpu.rb +27 -0
  10. data/lib/libv8/fpic-on-linux-amd64.patch +13 -0
  11. data/lib/libv8/v8/.gitignore +35 -0
  12. data/lib/libv8/v8/AUTHORS +44 -0
  13. data/lib/libv8/v8/ChangeLog +2839 -0
  14. data/lib/libv8/v8/LICENSE +52 -0
  15. data/lib/libv8/v8/LICENSE.strongtalk +29 -0
  16. data/lib/libv8/v8/LICENSE.v8 +26 -0
  17. data/lib/libv8/v8/LICENSE.valgrind +45 -0
  18. data/lib/libv8/v8/SConstruct +1478 -0
  19. data/lib/libv8/v8/build/README.txt +49 -0
  20. data/lib/libv8/v8/build/all.gyp +18 -0
  21. data/lib/libv8/v8/build/armu.gypi +32 -0
  22. data/lib/libv8/v8/build/common.gypi +144 -0
  23. data/lib/libv8/v8/build/gyp_v8 +145 -0
  24. data/lib/libv8/v8/include/v8-debug.h +395 -0
  25. data/lib/libv8/v8/include/v8-preparser.h +117 -0
  26. data/lib/libv8/v8/include/v8-profiler.h +505 -0
  27. data/lib/libv8/v8/include/v8-testing.h +104 -0
  28. data/lib/libv8/v8/include/v8.h +4124 -0
  29. data/lib/libv8/v8/include/v8stdint.h +53 -0
  30. data/lib/libv8/v8/preparser/SConscript +38 -0
  31. data/lib/libv8/v8/preparser/preparser-process.cc +379 -0
  32. data/lib/libv8/v8/src/SConscript +368 -0
  33. data/lib/libv8/v8/src/accessors.cc +767 -0
  34. data/lib/libv8/v8/src/accessors.h +123 -0
  35. data/lib/libv8/v8/src/allocation-inl.h +49 -0
  36. data/lib/libv8/v8/src/allocation.cc +122 -0
  37. data/lib/libv8/v8/src/allocation.h +143 -0
  38. data/lib/libv8/v8/src/api.cc +5845 -0
  39. data/lib/libv8/v8/src/api.h +574 -0
  40. data/lib/libv8/v8/src/apinatives.js +110 -0
  41. data/lib/libv8/v8/src/apiutils.h +73 -0
  42. data/lib/libv8/v8/src/arguments.h +118 -0
  43. data/lib/libv8/v8/src/arm/assembler-arm-inl.h +353 -0
  44. data/lib/libv8/v8/src/arm/assembler-arm.cc +2661 -0
  45. data/lib/libv8/v8/src/arm/assembler-arm.h +1375 -0
  46. data/lib/libv8/v8/src/arm/builtins-arm.cc +1658 -0
  47. data/lib/libv8/v8/src/arm/code-stubs-arm.cc +6398 -0
  48. data/lib/libv8/v8/src/arm/code-stubs-arm.h +673 -0
  49. data/lib/libv8/v8/src/arm/codegen-arm.cc +52 -0
  50. data/lib/libv8/v8/src/arm/codegen-arm.h +91 -0
  51. data/lib/libv8/v8/src/arm/constants-arm.cc +152 -0
  52. data/lib/libv8/v8/src/arm/constants-arm.h +775 -0
  53. data/lib/libv8/v8/src/arm/cpu-arm.cc +120 -0
  54. data/lib/libv8/v8/src/arm/debug-arm.cc +317 -0
  55. data/lib/libv8/v8/src/arm/deoptimizer-arm.cc +754 -0
  56. data/lib/libv8/v8/src/arm/disasm-arm.cc +1506 -0
  57. data/lib/libv8/v8/src/arm/frames-arm.cc +45 -0
  58. data/lib/libv8/v8/src/arm/frames-arm.h +168 -0
  59. data/lib/libv8/v8/src/arm/full-codegen-arm.cc +4375 -0
  60. data/lib/libv8/v8/src/arm/ic-arm.cc +1562 -0
  61. data/lib/libv8/v8/src/arm/lithium-arm.cc +2206 -0
  62. data/lib/libv8/v8/src/arm/lithium-arm.h +2348 -0
  63. data/lib/libv8/v8/src/arm/lithium-codegen-arm.cc +4526 -0
  64. data/lib/libv8/v8/src/arm/lithium-codegen-arm.h +403 -0
  65. data/lib/libv8/v8/src/arm/lithium-gap-resolver-arm.cc +305 -0
  66. data/lib/libv8/v8/src/arm/lithium-gap-resolver-arm.h +84 -0
  67. data/lib/libv8/v8/src/arm/macro-assembler-arm.cc +3163 -0
  68. data/lib/libv8/v8/src/arm/macro-assembler-arm.h +1126 -0
  69. data/lib/libv8/v8/src/arm/regexp-macro-assembler-arm.cc +1287 -0
  70. data/lib/libv8/v8/src/arm/regexp-macro-assembler-arm.h +253 -0
  71. data/lib/libv8/v8/src/arm/simulator-arm.cc +3424 -0
  72. data/lib/libv8/v8/src/arm/simulator-arm.h +431 -0
  73. data/lib/libv8/v8/src/arm/stub-cache-arm.cc +4243 -0
  74. data/lib/libv8/v8/src/array.js +1366 -0
  75. data/lib/libv8/v8/src/assembler.cc +1207 -0
  76. data/lib/libv8/v8/src/assembler.h +858 -0
  77. data/lib/libv8/v8/src/ast-inl.h +112 -0
  78. data/lib/libv8/v8/src/ast.cc +1146 -0
  79. data/lib/libv8/v8/src/ast.h +2188 -0
  80. data/lib/libv8/v8/src/atomicops.h +167 -0
  81. data/lib/libv8/v8/src/atomicops_internals_arm_gcc.h +145 -0
  82. data/lib/libv8/v8/src/atomicops_internals_mips_gcc.h +169 -0
  83. data/lib/libv8/v8/src/atomicops_internals_x86_gcc.cc +133 -0
  84. data/lib/libv8/v8/src/atomicops_internals_x86_gcc.h +287 -0
  85. data/lib/libv8/v8/src/atomicops_internals_x86_macosx.h +301 -0
  86. data/lib/libv8/v8/src/atomicops_internals_x86_msvc.h +203 -0
  87. data/lib/libv8/v8/src/bignum-dtoa.cc +655 -0
  88. data/lib/libv8/v8/src/bignum-dtoa.h +81 -0
  89. data/lib/libv8/v8/src/bignum.cc +768 -0
  90. data/lib/libv8/v8/src/bignum.h +140 -0
  91. data/lib/libv8/v8/src/bootstrapper.cc +2184 -0
  92. data/lib/libv8/v8/src/bootstrapper.h +188 -0
  93. data/lib/libv8/v8/src/builtins.cc +1707 -0
  94. data/lib/libv8/v8/src/builtins.h +371 -0
  95. data/lib/libv8/v8/src/bytecodes-irregexp.h +105 -0
  96. data/lib/libv8/v8/src/cached-powers.cc +177 -0
  97. data/lib/libv8/v8/src/cached-powers.h +65 -0
  98. data/lib/libv8/v8/src/char-predicates-inl.h +94 -0
  99. data/lib/libv8/v8/src/char-predicates.h +67 -0
  100. data/lib/libv8/v8/src/checks.cc +110 -0
  101. data/lib/libv8/v8/src/checks.h +296 -0
  102. data/lib/libv8/v8/src/circular-queue-inl.h +53 -0
  103. data/lib/libv8/v8/src/circular-queue.cc +122 -0
  104. data/lib/libv8/v8/src/circular-queue.h +103 -0
  105. data/lib/libv8/v8/src/code-stubs.cc +267 -0
  106. data/lib/libv8/v8/src/code-stubs.h +1011 -0
  107. data/lib/libv8/v8/src/code.h +70 -0
  108. data/lib/libv8/v8/src/codegen.cc +231 -0
  109. data/lib/libv8/v8/src/codegen.h +84 -0
  110. data/lib/libv8/v8/src/compilation-cache.cc +540 -0
  111. data/lib/libv8/v8/src/compilation-cache.h +287 -0
  112. data/lib/libv8/v8/src/compiler.cc +786 -0
  113. data/lib/libv8/v8/src/compiler.h +312 -0
  114. data/lib/libv8/v8/src/contexts.cc +347 -0
  115. data/lib/libv8/v8/src/contexts.h +391 -0
  116. data/lib/libv8/v8/src/conversions-inl.h +106 -0
  117. data/lib/libv8/v8/src/conversions.cc +1131 -0
  118. data/lib/libv8/v8/src/conversions.h +135 -0
  119. data/lib/libv8/v8/src/counters.cc +93 -0
  120. data/lib/libv8/v8/src/counters.h +254 -0
  121. data/lib/libv8/v8/src/cpu-profiler-inl.h +101 -0
  122. data/lib/libv8/v8/src/cpu-profiler.cc +609 -0
  123. data/lib/libv8/v8/src/cpu-profiler.h +302 -0
  124. data/lib/libv8/v8/src/cpu.h +69 -0
  125. data/lib/libv8/v8/src/d8-debug.cc +367 -0
  126. data/lib/libv8/v8/src/d8-debug.h +158 -0
  127. data/lib/libv8/v8/src/d8-posix.cc +695 -0
  128. data/lib/libv8/v8/src/d8-readline.cc +130 -0
  129. data/lib/libv8/v8/src/d8-windows.cc +42 -0
  130. data/lib/libv8/v8/src/d8.cc +803 -0
  131. data/lib/libv8/v8/src/d8.gyp +91 -0
  132. data/lib/libv8/v8/src/d8.h +235 -0
  133. data/lib/libv8/v8/src/d8.js +2798 -0
  134. data/lib/libv8/v8/src/data-flow.cc +66 -0
  135. data/lib/libv8/v8/src/data-flow.h +205 -0
  136. data/lib/libv8/v8/src/date.js +1103 -0
  137. data/lib/libv8/v8/src/dateparser-inl.h +127 -0
  138. data/lib/libv8/v8/src/dateparser.cc +178 -0
  139. data/lib/libv8/v8/src/dateparser.h +266 -0
  140. data/lib/libv8/v8/src/debug-agent.cc +447 -0
  141. data/lib/libv8/v8/src/debug-agent.h +129 -0
  142. data/lib/libv8/v8/src/debug-debugger.js +2569 -0
  143. data/lib/libv8/v8/src/debug.cc +3165 -0
  144. data/lib/libv8/v8/src/debug.h +1057 -0
  145. data/lib/libv8/v8/src/deoptimizer.cc +1256 -0
  146. data/lib/libv8/v8/src/deoptimizer.h +602 -0
  147. data/lib/libv8/v8/src/disasm.h +80 -0
  148. data/lib/libv8/v8/src/disassembler.cc +343 -0
  149. data/lib/libv8/v8/src/disassembler.h +58 -0
  150. data/lib/libv8/v8/src/diy-fp.cc +58 -0
  151. data/lib/libv8/v8/src/diy-fp.h +117 -0
  152. data/lib/libv8/v8/src/double.h +238 -0
  153. data/lib/libv8/v8/src/dtoa.cc +103 -0
  154. data/lib/libv8/v8/src/dtoa.h +85 -0
  155. data/lib/libv8/v8/src/execution.cc +849 -0
  156. data/lib/libv8/v8/src/execution.h +297 -0
  157. data/lib/libv8/v8/src/extensions/experimental/break-iterator.cc +250 -0
  158. data/lib/libv8/v8/src/extensions/experimental/break-iterator.h +89 -0
  159. data/lib/libv8/v8/src/extensions/experimental/collator.cc +218 -0
  160. data/lib/libv8/v8/src/extensions/experimental/collator.h +69 -0
  161. data/lib/libv8/v8/src/extensions/experimental/experimental.gyp +94 -0
  162. data/lib/libv8/v8/src/extensions/experimental/i18n-extension.cc +78 -0
  163. data/lib/libv8/v8/src/extensions/experimental/i18n-extension.h +54 -0
  164. data/lib/libv8/v8/src/extensions/experimental/i18n-locale.cc +112 -0
  165. data/lib/libv8/v8/src/extensions/experimental/i18n-locale.h +60 -0
  166. data/lib/libv8/v8/src/extensions/experimental/i18n-utils.cc +43 -0
  167. data/lib/libv8/v8/src/extensions/experimental/i18n-utils.h +49 -0
  168. data/lib/libv8/v8/src/extensions/experimental/i18n.js +180 -0
  169. data/lib/libv8/v8/src/extensions/experimental/language-matcher.cc +251 -0
  170. data/lib/libv8/v8/src/extensions/experimental/language-matcher.h +95 -0
  171. data/lib/libv8/v8/src/extensions/externalize-string-extension.cc +141 -0
  172. data/lib/libv8/v8/src/extensions/externalize-string-extension.h +50 -0
  173. data/lib/libv8/v8/src/extensions/gc-extension.cc +58 -0
  174. data/lib/libv8/v8/src/extensions/gc-extension.h +49 -0
  175. data/lib/libv8/v8/src/factory.cc +1222 -0
  176. data/lib/libv8/v8/src/factory.h +442 -0
  177. data/lib/libv8/v8/src/fast-dtoa.cc +736 -0
  178. data/lib/libv8/v8/src/fast-dtoa.h +83 -0
  179. data/lib/libv8/v8/src/fixed-dtoa.cc +405 -0
  180. data/lib/libv8/v8/src/fixed-dtoa.h +55 -0
  181. data/lib/libv8/v8/src/flag-definitions.h +560 -0
  182. data/lib/libv8/v8/src/flags.cc +551 -0
  183. data/lib/libv8/v8/src/flags.h +79 -0
  184. data/lib/libv8/v8/src/frames-inl.h +247 -0
  185. data/lib/libv8/v8/src/frames.cc +1243 -0
  186. data/lib/libv8/v8/src/frames.h +870 -0
  187. data/lib/libv8/v8/src/full-codegen.cc +1374 -0
  188. data/lib/libv8/v8/src/full-codegen.h +771 -0
  189. data/lib/libv8/v8/src/func-name-inferrer.cc +92 -0
  190. data/lib/libv8/v8/src/func-name-inferrer.h +111 -0
  191. data/lib/libv8/v8/src/gdb-jit.cc +1555 -0
  192. data/lib/libv8/v8/src/gdb-jit.h +143 -0
  193. data/lib/libv8/v8/src/global-handles.cc +665 -0
  194. data/lib/libv8/v8/src/global-handles.h +284 -0
  195. data/lib/libv8/v8/src/globals.h +325 -0
  196. data/lib/libv8/v8/src/handles-inl.h +177 -0
  197. data/lib/libv8/v8/src/handles.cc +987 -0
  198. data/lib/libv8/v8/src/handles.h +382 -0
  199. data/lib/libv8/v8/src/hashmap.cc +230 -0
  200. data/lib/libv8/v8/src/hashmap.h +123 -0
  201. data/lib/libv8/v8/src/heap-inl.h +704 -0
  202. data/lib/libv8/v8/src/heap-profiler.cc +1173 -0
  203. data/lib/libv8/v8/src/heap-profiler.h +397 -0
  204. data/lib/libv8/v8/src/heap.cc +5930 -0
  205. data/lib/libv8/v8/src/heap.h +2268 -0
  206. data/lib/libv8/v8/src/hydrogen-instructions.cc +1769 -0
  207. data/lib/libv8/v8/src/hydrogen-instructions.h +3971 -0
  208. data/lib/libv8/v8/src/hydrogen.cc +6239 -0
  209. data/lib/libv8/v8/src/hydrogen.h +1202 -0
  210. data/lib/libv8/v8/src/ia32/assembler-ia32-inl.h +446 -0
  211. data/lib/libv8/v8/src/ia32/assembler-ia32.cc +2487 -0
  212. data/lib/libv8/v8/src/ia32/assembler-ia32.h +1144 -0
  213. data/lib/libv8/v8/src/ia32/builtins-ia32.cc +1621 -0
  214. data/lib/libv8/v8/src/ia32/code-stubs-ia32.cc +6198 -0
  215. data/lib/libv8/v8/src/ia32/code-stubs-ia32.h +517 -0
  216. data/lib/libv8/v8/src/ia32/codegen-ia32.cc +265 -0
  217. data/lib/libv8/v8/src/ia32/codegen-ia32.h +79 -0
  218. data/lib/libv8/v8/src/ia32/cpu-ia32.cc +88 -0
  219. data/lib/libv8/v8/src/ia32/debug-ia32.cc +312 -0
  220. data/lib/libv8/v8/src/ia32/deoptimizer-ia32.cc +774 -0
  221. data/lib/libv8/v8/src/ia32/disasm-ia32.cc +1628 -0
  222. data/lib/libv8/v8/src/ia32/frames-ia32.cc +45 -0
  223. data/lib/libv8/v8/src/ia32/frames-ia32.h +142 -0
  224. data/lib/libv8/v8/src/ia32/full-codegen-ia32.cc +4338 -0
  225. data/lib/libv8/v8/src/ia32/ic-ia32.cc +1597 -0
  226. data/lib/libv8/v8/src/ia32/lithium-codegen-ia32.cc +4461 -0
  227. data/lib/libv8/v8/src/ia32/lithium-codegen-ia32.h +375 -0
  228. data/lib/libv8/v8/src/ia32/lithium-gap-resolver-ia32.cc +475 -0
  229. data/lib/libv8/v8/src/ia32/lithium-gap-resolver-ia32.h +110 -0
  230. data/lib/libv8/v8/src/ia32/lithium-ia32.cc +2261 -0
  231. data/lib/libv8/v8/src/ia32/lithium-ia32.h +2396 -0
  232. data/lib/libv8/v8/src/ia32/macro-assembler-ia32.cc +2136 -0
  233. data/lib/libv8/v8/src/ia32/macro-assembler-ia32.h +775 -0
  234. data/lib/libv8/v8/src/ia32/regexp-macro-assembler-ia32.cc +1263 -0
  235. data/lib/libv8/v8/src/ia32/regexp-macro-assembler-ia32.h +216 -0
  236. data/lib/libv8/v8/src/ia32/simulator-ia32.cc +30 -0
  237. data/lib/libv8/v8/src/ia32/simulator-ia32.h +74 -0
  238. data/lib/libv8/v8/src/ia32/stub-cache-ia32.cc +3847 -0
  239. data/lib/libv8/v8/src/ic-inl.h +130 -0
  240. data/lib/libv8/v8/src/ic.cc +2577 -0
  241. data/lib/libv8/v8/src/ic.h +736 -0
  242. data/lib/libv8/v8/src/inspector.cc +63 -0
  243. data/lib/libv8/v8/src/inspector.h +62 -0
  244. data/lib/libv8/v8/src/interpreter-irregexp.cc +659 -0
  245. data/lib/libv8/v8/src/interpreter-irregexp.h +49 -0
  246. data/lib/libv8/v8/src/isolate-inl.h +50 -0
  247. data/lib/libv8/v8/src/isolate.cc +1869 -0
  248. data/lib/libv8/v8/src/isolate.h +1382 -0
  249. data/lib/libv8/v8/src/json-parser.cc +504 -0
  250. data/lib/libv8/v8/src/json-parser.h +161 -0
  251. data/lib/libv8/v8/src/json.js +342 -0
  252. data/lib/libv8/v8/src/jsregexp.cc +5385 -0
  253. data/lib/libv8/v8/src/jsregexp.h +1492 -0
  254. data/lib/libv8/v8/src/list-inl.h +212 -0
  255. data/lib/libv8/v8/src/list.h +174 -0
  256. data/lib/libv8/v8/src/lithium-allocator-inl.h +142 -0
  257. data/lib/libv8/v8/src/lithium-allocator.cc +2123 -0
  258. data/lib/libv8/v8/src/lithium-allocator.h +630 -0
  259. data/lib/libv8/v8/src/lithium.cc +190 -0
  260. data/lib/libv8/v8/src/lithium.h +597 -0
  261. data/lib/libv8/v8/src/liveedit-debugger.js +1082 -0
  262. data/lib/libv8/v8/src/liveedit.cc +1691 -0
  263. data/lib/libv8/v8/src/liveedit.h +180 -0
  264. data/lib/libv8/v8/src/liveobjectlist-inl.h +126 -0
  265. data/lib/libv8/v8/src/liveobjectlist.cc +2589 -0
  266. data/lib/libv8/v8/src/liveobjectlist.h +322 -0
  267. data/lib/libv8/v8/src/log-inl.h +59 -0
  268. data/lib/libv8/v8/src/log-utils.cc +428 -0
  269. data/lib/libv8/v8/src/log-utils.h +231 -0
  270. data/lib/libv8/v8/src/log.cc +1993 -0
  271. data/lib/libv8/v8/src/log.h +476 -0
  272. data/lib/libv8/v8/src/macro-assembler.h +120 -0
  273. data/lib/libv8/v8/src/macros.py +178 -0
  274. data/lib/libv8/v8/src/mark-compact.cc +3143 -0
  275. data/lib/libv8/v8/src/mark-compact.h +506 -0
  276. data/lib/libv8/v8/src/math.js +264 -0
  277. data/lib/libv8/v8/src/messages.cc +179 -0
  278. data/lib/libv8/v8/src/messages.h +113 -0
  279. data/lib/libv8/v8/src/messages.js +1096 -0
  280. data/lib/libv8/v8/src/mips/assembler-mips-inl.h +312 -0
  281. data/lib/libv8/v8/src/mips/assembler-mips.cc +1960 -0
  282. data/lib/libv8/v8/src/mips/assembler-mips.h +1138 -0
  283. data/lib/libv8/v8/src/mips/builtins-mips.cc +1628 -0
  284. data/lib/libv8/v8/src/mips/code-stubs-mips.cc +6656 -0
  285. data/lib/libv8/v8/src/mips/code-stubs-mips.h +682 -0
  286. data/lib/libv8/v8/src/mips/codegen-mips.cc +52 -0
  287. data/lib/libv8/v8/src/mips/codegen-mips.h +98 -0
  288. data/lib/libv8/v8/src/mips/constants-mips.cc +352 -0
  289. data/lib/libv8/v8/src/mips/constants-mips.h +739 -0
  290. data/lib/libv8/v8/src/mips/cpu-mips.cc +96 -0
  291. data/lib/libv8/v8/src/mips/debug-mips.cc +308 -0
  292. data/lib/libv8/v8/src/mips/deoptimizer-mips.cc +91 -0
  293. data/lib/libv8/v8/src/mips/disasm-mips.cc +1050 -0
  294. data/lib/libv8/v8/src/mips/frames-mips.cc +47 -0
  295. data/lib/libv8/v8/src/mips/frames-mips.h +219 -0
  296. data/lib/libv8/v8/src/mips/full-codegen-mips.cc +4388 -0
  297. data/lib/libv8/v8/src/mips/ic-mips.cc +1580 -0
  298. data/lib/libv8/v8/src/mips/lithium-codegen-mips.h +65 -0
  299. data/lib/libv8/v8/src/mips/lithium-mips.h +307 -0
  300. data/lib/libv8/v8/src/mips/macro-assembler-mips.cc +4056 -0
  301. data/lib/libv8/v8/src/mips/macro-assembler-mips.h +1214 -0
  302. data/lib/libv8/v8/src/mips/regexp-macro-assembler-mips.cc +1251 -0
  303. data/lib/libv8/v8/src/mips/regexp-macro-assembler-mips.h +252 -0
  304. data/lib/libv8/v8/src/mips/simulator-mips.cc +2621 -0
  305. data/lib/libv8/v8/src/mips/simulator-mips.h +401 -0
  306. data/lib/libv8/v8/src/mips/stub-cache-mips.cc +4285 -0
  307. data/lib/libv8/v8/src/mirror-debugger.js +2382 -0
  308. data/lib/libv8/v8/src/mksnapshot.cc +328 -0
  309. data/lib/libv8/v8/src/natives.h +64 -0
  310. data/lib/libv8/v8/src/objects-debug.cc +738 -0
  311. data/lib/libv8/v8/src/objects-inl.h +4323 -0
  312. data/lib/libv8/v8/src/objects-printer.cc +829 -0
  313. data/lib/libv8/v8/src/objects-visiting.cc +148 -0
  314. data/lib/libv8/v8/src/objects-visiting.h +424 -0
  315. data/lib/libv8/v8/src/objects.cc +10585 -0
  316. data/lib/libv8/v8/src/objects.h +6838 -0
  317. data/lib/libv8/v8/src/parser.cc +4997 -0
  318. data/lib/libv8/v8/src/parser.h +765 -0
  319. data/lib/libv8/v8/src/platform-cygwin.cc +779 -0
  320. data/lib/libv8/v8/src/platform-freebsd.cc +826 -0
  321. data/lib/libv8/v8/src/platform-linux.cc +1149 -0
  322. data/lib/libv8/v8/src/platform-macos.cc +830 -0
  323. data/lib/libv8/v8/src/platform-nullos.cc +479 -0
  324. data/lib/libv8/v8/src/platform-openbsd.cc +640 -0
  325. data/lib/libv8/v8/src/platform-posix.cc +424 -0
  326. data/lib/libv8/v8/src/platform-solaris.cc +762 -0
  327. data/lib/libv8/v8/src/platform-tls-mac.h +62 -0
  328. data/lib/libv8/v8/src/platform-tls-win32.h +62 -0
  329. data/lib/libv8/v8/src/platform-tls.h +50 -0
  330. data/lib/libv8/v8/src/platform-win32.cc +2021 -0
  331. data/lib/libv8/v8/src/platform.h +667 -0
  332. data/lib/libv8/v8/src/preparse-data-format.h +62 -0
  333. data/lib/libv8/v8/src/preparse-data.cc +183 -0
  334. data/lib/libv8/v8/src/preparse-data.h +225 -0
  335. data/lib/libv8/v8/src/preparser-api.cc +220 -0
  336. data/lib/libv8/v8/src/preparser.cc +1450 -0
  337. data/lib/libv8/v8/src/preparser.h +493 -0
  338. data/lib/libv8/v8/src/prettyprinter.cc +1493 -0
  339. data/lib/libv8/v8/src/prettyprinter.h +223 -0
  340. data/lib/libv8/v8/src/profile-generator-inl.h +128 -0
  341. data/lib/libv8/v8/src/profile-generator.cc +3098 -0
  342. data/lib/libv8/v8/src/profile-generator.h +1126 -0
  343. data/lib/libv8/v8/src/property.cc +105 -0
  344. data/lib/libv8/v8/src/property.h +365 -0
  345. data/lib/libv8/v8/src/proxy.js +83 -0
  346. data/lib/libv8/v8/src/regexp-macro-assembler-irregexp-inl.h +78 -0
  347. data/lib/libv8/v8/src/regexp-macro-assembler-irregexp.cc +471 -0
  348. data/lib/libv8/v8/src/regexp-macro-assembler-irregexp.h +142 -0
  349. data/lib/libv8/v8/src/regexp-macro-assembler-tracer.cc +373 -0
  350. data/lib/libv8/v8/src/regexp-macro-assembler-tracer.h +104 -0
  351. data/lib/libv8/v8/src/regexp-macro-assembler.cc +267 -0
  352. data/lib/libv8/v8/src/regexp-macro-assembler.h +243 -0
  353. data/lib/libv8/v8/src/regexp-stack.cc +111 -0
  354. data/lib/libv8/v8/src/regexp-stack.h +147 -0
  355. data/lib/libv8/v8/src/regexp.js +483 -0
  356. data/lib/libv8/v8/src/rewriter.cc +360 -0
  357. data/lib/libv8/v8/src/rewriter.h +50 -0
  358. data/lib/libv8/v8/src/runtime-profiler.cc +489 -0
  359. data/lib/libv8/v8/src/runtime-profiler.h +201 -0
  360. data/lib/libv8/v8/src/runtime.cc +12227 -0
  361. data/lib/libv8/v8/src/runtime.h +652 -0
  362. data/lib/libv8/v8/src/runtime.js +649 -0
  363. data/lib/libv8/v8/src/safepoint-table.cc +256 -0
  364. data/lib/libv8/v8/src/safepoint-table.h +270 -0
  365. data/lib/libv8/v8/src/scanner-base.cc +952 -0
  366. data/lib/libv8/v8/src/scanner-base.h +670 -0
  367. data/lib/libv8/v8/src/scanner.cc +345 -0
  368. data/lib/libv8/v8/src/scanner.h +146 -0
  369. data/lib/libv8/v8/src/scopeinfo.cc +646 -0
  370. data/lib/libv8/v8/src/scopeinfo.h +254 -0
  371. data/lib/libv8/v8/src/scopes.cc +1150 -0
  372. data/lib/libv8/v8/src/scopes.h +507 -0
  373. data/lib/libv8/v8/src/serialize.cc +1574 -0
  374. data/lib/libv8/v8/src/serialize.h +589 -0
  375. data/lib/libv8/v8/src/shell.h +55 -0
  376. data/lib/libv8/v8/src/simulator.h +43 -0
  377. data/lib/libv8/v8/src/small-pointer-list.h +163 -0
  378. data/lib/libv8/v8/src/smart-pointer.h +109 -0
  379. data/lib/libv8/v8/src/snapshot-common.cc +83 -0
  380. data/lib/libv8/v8/src/snapshot-empty.cc +54 -0
  381. data/lib/libv8/v8/src/snapshot.h +91 -0
  382. data/lib/libv8/v8/src/spaces-inl.h +529 -0
  383. data/lib/libv8/v8/src/spaces.cc +3145 -0
  384. data/lib/libv8/v8/src/spaces.h +2369 -0
  385. data/lib/libv8/v8/src/splay-tree-inl.h +310 -0
  386. data/lib/libv8/v8/src/splay-tree.h +205 -0
  387. data/lib/libv8/v8/src/string-search.cc +41 -0
  388. data/lib/libv8/v8/src/string-search.h +568 -0
  389. data/lib/libv8/v8/src/string-stream.cc +592 -0
  390. data/lib/libv8/v8/src/string-stream.h +191 -0
  391. data/lib/libv8/v8/src/string.js +994 -0
  392. data/lib/libv8/v8/src/strtod.cc +440 -0
  393. data/lib/libv8/v8/src/strtod.h +40 -0
  394. data/lib/libv8/v8/src/stub-cache.cc +1965 -0
  395. data/lib/libv8/v8/src/stub-cache.h +924 -0
  396. data/lib/libv8/v8/src/third_party/valgrind/valgrind.h +3925 -0
  397. data/lib/libv8/v8/src/token.cc +63 -0
  398. data/lib/libv8/v8/src/token.h +288 -0
  399. data/lib/libv8/v8/src/type-info.cc +507 -0
  400. data/lib/libv8/v8/src/type-info.h +272 -0
  401. data/lib/libv8/v8/src/unbound-queue-inl.h +95 -0
  402. data/lib/libv8/v8/src/unbound-queue.h +69 -0
  403. data/lib/libv8/v8/src/unicode-inl.h +238 -0
  404. data/lib/libv8/v8/src/unicode.cc +1624 -0
  405. data/lib/libv8/v8/src/unicode.h +280 -0
  406. data/lib/libv8/v8/src/uri.js +408 -0
  407. data/lib/libv8/v8/src/utils-inl.h +48 -0
  408. data/lib/libv8/v8/src/utils.cc +371 -0
  409. data/lib/libv8/v8/src/utils.h +800 -0
  410. data/lib/libv8/v8/src/v8-counters.cc +62 -0
  411. data/lib/libv8/v8/src/v8-counters.h +314 -0
  412. data/lib/libv8/v8/src/v8.cc +213 -0
  413. data/lib/libv8/v8/src/v8.h +131 -0
  414. data/lib/libv8/v8/src/v8checks.h +64 -0
  415. data/lib/libv8/v8/src/v8dll-main.cc +44 -0
  416. data/lib/libv8/v8/src/v8globals.h +512 -0
  417. data/lib/libv8/v8/src/v8memory.h +82 -0
  418. data/lib/libv8/v8/src/v8natives.js +1310 -0
  419. data/lib/libv8/v8/src/v8preparserdll-main.cc +39 -0
  420. data/lib/libv8/v8/src/v8threads.cc +464 -0
  421. data/lib/libv8/v8/src/v8threads.h +165 -0
  422. data/lib/libv8/v8/src/v8utils.h +319 -0
  423. data/lib/libv8/v8/src/variables.cc +114 -0
  424. data/lib/libv8/v8/src/variables.h +167 -0
  425. data/lib/libv8/v8/src/version.cc +116 -0
  426. data/lib/libv8/v8/src/version.h +68 -0
  427. data/lib/libv8/v8/src/vm-state-inl.h +138 -0
  428. data/lib/libv8/v8/src/vm-state.h +71 -0
  429. data/lib/libv8/v8/src/win32-headers.h +96 -0
  430. data/lib/libv8/v8/src/x64/assembler-x64-inl.h +462 -0
  431. data/lib/libv8/v8/src/x64/assembler-x64.cc +3027 -0
  432. data/lib/libv8/v8/src/x64/assembler-x64.h +1633 -0
  433. data/lib/libv8/v8/src/x64/builtins-x64.cc +1520 -0
  434. data/lib/libv8/v8/src/x64/code-stubs-x64.cc +5132 -0
  435. data/lib/libv8/v8/src/x64/code-stubs-x64.h +514 -0
  436. data/lib/libv8/v8/src/x64/codegen-x64.cc +146 -0
  437. data/lib/libv8/v8/src/x64/codegen-x64.h +76 -0
  438. data/lib/libv8/v8/src/x64/cpu-x64.cc +88 -0
  439. data/lib/libv8/v8/src/x64/debug-x64.cc +319 -0
  440. data/lib/libv8/v8/src/x64/deoptimizer-x64.cc +815 -0
  441. data/lib/libv8/v8/src/x64/disasm-x64.cc +1832 -0
  442. data/lib/libv8/v8/src/x64/frames-x64.cc +45 -0
  443. data/lib/libv8/v8/src/x64/frames-x64.h +130 -0
  444. data/lib/libv8/v8/src/x64/full-codegen-x64.cc +4318 -0
  445. data/lib/libv8/v8/src/x64/ic-x64.cc +1608 -0
  446. data/lib/libv8/v8/src/x64/lithium-codegen-x64.cc +4267 -0
  447. data/lib/libv8/v8/src/x64/lithium-codegen-x64.h +367 -0
  448. data/lib/libv8/v8/src/x64/lithium-gap-resolver-x64.cc +320 -0
  449. data/lib/libv8/v8/src/x64/lithium-gap-resolver-x64.h +74 -0
  450. data/lib/libv8/v8/src/x64/lithium-x64.cc +2202 -0
  451. data/lib/libv8/v8/src/x64/lithium-x64.h +2333 -0
  452. data/lib/libv8/v8/src/x64/macro-assembler-x64.cc +3745 -0
  453. data/lib/libv8/v8/src/x64/macro-assembler-x64.h +1290 -0
  454. data/lib/libv8/v8/src/x64/regexp-macro-assembler-x64.cc +1398 -0
  455. data/lib/libv8/v8/src/x64/regexp-macro-assembler-x64.h +282 -0
  456. data/lib/libv8/v8/src/x64/simulator-x64.cc +27 -0
  457. data/lib/libv8/v8/src/x64/simulator-x64.h +72 -0
  458. data/lib/libv8/v8/src/x64/stub-cache-x64.cc +3610 -0
  459. data/lib/libv8/v8/src/zone-inl.h +140 -0
  460. data/lib/libv8/v8/src/zone.cc +196 -0
  461. data/lib/libv8/v8/src/zone.h +240 -0
  462. data/lib/libv8/v8/tools/codemap.js +265 -0
  463. data/lib/libv8/v8/tools/consarray.js +93 -0
  464. data/lib/libv8/v8/tools/csvparser.js +78 -0
  465. data/lib/libv8/v8/tools/disasm.py +92 -0
  466. data/lib/libv8/v8/tools/freebsd-tick-processor +10 -0
  467. data/lib/libv8/v8/tools/gc-nvp-trace-processor.py +342 -0
  468. data/lib/libv8/v8/tools/gcmole/README +62 -0
  469. data/lib/libv8/v8/tools/gcmole/gccause.lua +60 -0
  470. data/lib/libv8/v8/tools/gcmole/gcmole.cc +1261 -0
  471. data/lib/libv8/v8/tools/gcmole/gcmole.lua +378 -0
  472. data/lib/libv8/v8/tools/generate-ten-powers.scm +286 -0
  473. data/lib/libv8/v8/tools/grokdump.py +841 -0
  474. data/lib/libv8/v8/tools/gyp/v8.gyp +995 -0
  475. data/lib/libv8/v8/tools/js2c.py +364 -0
  476. data/lib/libv8/v8/tools/jsmin.py +280 -0
  477. data/lib/libv8/v8/tools/linux-tick-processor +35 -0
  478. data/lib/libv8/v8/tools/ll_prof.py +942 -0
  479. data/lib/libv8/v8/tools/logreader.js +185 -0
  480. data/lib/libv8/v8/tools/mac-nm +18 -0
  481. data/lib/libv8/v8/tools/mac-tick-processor +6 -0
  482. data/lib/libv8/v8/tools/oom_dump/README +31 -0
  483. data/lib/libv8/v8/tools/oom_dump/SConstruct +42 -0
  484. data/lib/libv8/v8/tools/oom_dump/oom_dump.cc +288 -0
  485. data/lib/libv8/v8/tools/presubmit.py +305 -0
  486. data/lib/libv8/v8/tools/process-heap-prof.py +120 -0
  487. data/lib/libv8/v8/tools/profile.js +751 -0
  488. data/lib/libv8/v8/tools/profile_view.js +219 -0
  489. data/lib/libv8/v8/tools/run-valgrind.py +77 -0
  490. data/lib/libv8/v8/tools/splaytree.js +316 -0
  491. data/lib/libv8/v8/tools/stats-viewer.py +468 -0
  492. data/lib/libv8/v8/tools/test.py +1510 -0
  493. data/lib/libv8/v8/tools/tickprocessor-driver.js +59 -0
  494. data/lib/libv8/v8/tools/tickprocessor.js +877 -0
  495. data/lib/libv8/v8/tools/utils.py +96 -0
  496. data/lib/libv8/v8/tools/visual_studio/README.txt +12 -0
  497. data/lib/libv8/v8/tools/windows-tick-processor.bat +30 -0
  498. data/lib/libv8/version.rb +5 -0
  499. data/libv8.gemspec +36 -0
  500. metadata +578 -0
@@ -0,0 +1,92 @@
1
+ // Copyright 2011 the V8 project authors. All rights reserved.
2
+ // Redistribution and use in source and binary forms, with or without
3
+ // modification, are permitted provided that the following conditions are
4
+ // met:
5
+ //
6
+ // * Redistributions of source code must retain the above copyright
7
+ // notice, this list of conditions and the following disclaimer.
8
+ // * Redistributions in binary form must reproduce the above
9
+ // copyright notice, this list of conditions and the following
10
+ // disclaimer in the documentation and/or other materials provided
11
+ // with the distribution.
12
+ // * Neither the name of Google Inc. nor the names of its
13
+ // contributors may be used to endorse or promote products derived
14
+ // from this software without specific prior written permission.
15
+ //
16
+ // THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS
17
+ // "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT
18
+ // LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR
19
+ // A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT
20
+ // OWNER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL,
21
+ // SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT
22
+ // LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE,
23
+ // DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY
24
+ // THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
25
+ // (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE
26
+ // OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
27
+
28
+ #include "v8.h"
29
+
30
+ #include "ast.h"
31
+ #include "func-name-inferrer.h"
32
+ #include "list-inl.h"
33
+
34
+ namespace v8 {
35
+ namespace internal {
36
+
37
+
38
+ void FuncNameInferrer::PushEnclosingName(Handle<String> name) {
39
+ // Enclosing name is a name of a constructor function. To check
40
+ // that it is really a constructor, we check that it is not empty
41
+ // and starts with a capital letter.
42
+ if (name->length() > 0 && Runtime::IsUpperCaseChar(
43
+ Isolate::Current()->runtime_state(), name->Get(0))) {
44
+ names_stack_.Add(name);
45
+ }
46
+ }
47
+
48
+
49
+ void FuncNameInferrer::PushLiteralName(Handle<String> name) {
50
+ if (IsOpen() && !HEAP->prototype_symbol()->Equals(*name)) {
51
+ names_stack_.Add(name);
52
+ }
53
+ }
54
+
55
+
56
+ void FuncNameInferrer::PushVariableName(Handle<String> name) {
57
+ if (IsOpen() && !HEAP->result_symbol()->Equals(*name)) {
58
+ names_stack_.Add(name);
59
+ }
60
+ }
61
+
62
+
63
+ Handle<String> FuncNameInferrer::MakeNameFromStack() {
64
+ if (names_stack_.is_empty()) {
65
+ return FACTORY->empty_string();
66
+ } else {
67
+ return MakeNameFromStackHelper(1, names_stack_.at(0));
68
+ }
69
+ }
70
+
71
+
72
+ Handle<String> FuncNameInferrer::MakeNameFromStackHelper(int pos,
73
+ Handle<String> prev) {
74
+ if (pos >= names_stack_.length()) {
75
+ return prev;
76
+ } else {
77
+ Handle<String> curr = FACTORY->NewConsString(dot_, names_stack_.at(pos));
78
+ return MakeNameFromStackHelper(pos + 1, FACTORY->NewConsString(prev, curr));
79
+ }
80
+ }
81
+
82
+
83
+ void FuncNameInferrer::InferFunctionsNames() {
84
+ Handle<String> func_name = MakeNameFromStack();
85
+ for (int i = 0; i < funcs_to_infer_.length(); ++i) {
86
+ funcs_to_infer_[i]->set_inferred_name(func_name);
87
+ }
88
+ funcs_to_infer_.Rewind(0);
89
+ }
90
+
91
+
92
+ } } // namespace v8::internal
@@ -0,0 +1,111 @@
1
+ // Copyright 2006-2009 the V8 project authors. All rights reserved.
2
+ // Redistribution and use in source and binary forms, with or without
3
+ // modification, are permitted provided that the following conditions are
4
+ // met:
5
+ //
6
+ // * Redistributions of source code must retain the above copyright
7
+ // notice, this list of conditions and the following disclaimer.
8
+ // * Redistributions in binary form must reproduce the above
9
+ // copyright notice, this list of conditions and the following
10
+ // disclaimer in the documentation and/or other materials provided
11
+ // with the distribution.
12
+ // * Neither the name of Google Inc. nor the names of its
13
+ // contributors may be used to endorse or promote products derived
14
+ // from this software without specific prior written permission.
15
+ //
16
+ // THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS
17
+ // "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT
18
+ // LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR
19
+ // A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT
20
+ // OWNER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL,
21
+ // SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT
22
+ // LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE,
23
+ // DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY
24
+ // THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
25
+ // (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE
26
+ // OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
27
+
28
+ #ifndef V8_FUNC_NAME_INFERRER_H_
29
+ #define V8_FUNC_NAME_INFERRER_H_
30
+
31
+ namespace v8 {
32
+ namespace internal {
33
+
34
+ // FuncNameInferrer is a stateful class that is used to perform name
35
+ // inference for anonymous functions during static analysis of source code.
36
+ // Inference is performed in cases when an anonymous function is assigned
37
+ // to a variable or a property (see test-func-name-inference.cc for examples.)
38
+ //
39
+ // The basic idea is that during parsing of LHSs of certain expressions
40
+ // (assignments, declarations, object literals) we collect name strings,
41
+ // and during parsing of the RHS, a function literal can be collected. After
42
+ // parsing the RHS we can infer a name for function literals that do not have
43
+ // a name.
44
+ class FuncNameInferrer : public ZoneObject {
45
+ public:
46
+ FuncNameInferrer()
47
+ : entries_stack_(10),
48
+ names_stack_(5),
49
+ funcs_to_infer_(4),
50
+ dot_(FACTORY->NewStringFromAscii(CStrVector("."))) {
51
+ }
52
+
53
+ // Returns whether we have entered name collection state.
54
+ bool IsOpen() const { return !entries_stack_.is_empty(); }
55
+
56
+ // Pushes an enclosing the name of enclosing function onto names stack.
57
+ void PushEnclosingName(Handle<String> name);
58
+
59
+ // Enters name collection state.
60
+ void Enter() {
61
+ entries_stack_.Add(names_stack_.length());
62
+ }
63
+
64
+ // Pushes an encountered name onto names stack when in collection state.
65
+ void PushLiteralName(Handle<String> name);
66
+
67
+ void PushVariableName(Handle<String> name);
68
+
69
+ // Adds a function to infer name for.
70
+ void AddFunction(FunctionLiteral* func_to_infer) {
71
+ if (IsOpen()) {
72
+ funcs_to_infer_.Add(func_to_infer);
73
+ }
74
+ }
75
+
76
+ // Infers a function name and leaves names collection state.
77
+ void Infer() {
78
+ ASSERT(IsOpen());
79
+ if (!funcs_to_infer_.is_empty()) {
80
+ InferFunctionsNames();
81
+ }
82
+ }
83
+
84
+ // Infers a function name and leaves names collection state.
85
+ void Leave() {
86
+ ASSERT(IsOpen());
87
+ names_stack_.Rewind(entries_stack_.RemoveLast());
88
+ }
89
+
90
+ private:
91
+ // Constructs a full name in dotted notation from gathered names.
92
+ Handle<String> MakeNameFromStack();
93
+
94
+ // A helper function for MakeNameFromStack.
95
+ Handle<String> MakeNameFromStackHelper(int pos, Handle<String> prev);
96
+
97
+ // Performs name inferring for added functions.
98
+ void InferFunctionsNames();
99
+
100
+ ZoneList<int> entries_stack_;
101
+ ZoneList<Handle<String> > names_stack_;
102
+ ZoneList<FunctionLiteral*> funcs_to_infer_;
103
+ Handle<String> dot_;
104
+
105
+ DISALLOW_COPY_AND_ASSIGN(FuncNameInferrer);
106
+ };
107
+
108
+
109
+ } } // namespace v8::internal
110
+
111
+ #endif // V8_FUNC_NAME_INFERRER_H_
@@ -0,0 +1,1555 @@
1
+ // Copyright 2010 the V8 project authors. All rights reserved.
2
+ // Redistribution and use in source and binary forms, with or without
3
+ // modification, are permitted provided that the following conditions are
4
+ // met:
5
+ //
6
+ // * Redistributions of source code must retain the above copyright
7
+ // notice, this list of conditions and the following disclaimer.
8
+ // * Redistributions in binary form must reproduce the above
9
+ // copyright notice, this list of conditions and the following
10
+ // disclaimer in the documentation and/or other materials provided
11
+ // with the distribution.
12
+ // * Neither the name of Google Inc. nor the names of its
13
+ // contributors may be used to endorse or promote products derived
14
+ // from this software without specific prior written permission.
15
+ //
16
+ // THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS
17
+ // "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT
18
+ // LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR
19
+ // A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT
20
+ // OWNER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL,
21
+ // SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT
22
+ // LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE,
23
+ // DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY
24
+ // THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
25
+ // (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE
26
+ // OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
27
+
28
+ #ifdef ENABLE_GDB_JIT_INTERFACE
29
+ #include "v8.h"
30
+ #include "gdb-jit.h"
31
+
32
+ #include "bootstrapper.h"
33
+ #include "compiler.h"
34
+ #include "global-handles.h"
35
+ #include "messages.h"
36
+ #include "natives.h"
37
+
38
+ namespace v8 {
39
+ namespace internal {
40
+
41
+ class ELF;
42
+
43
+ class Writer BASE_EMBEDDED {
44
+ public:
45
+ explicit Writer(ELF* elf)
46
+ : elf_(elf),
47
+ position_(0),
48
+ capacity_(1024),
49
+ buffer_(reinterpret_cast<byte*>(malloc(capacity_))) {
50
+ }
51
+
52
+ ~Writer() {
53
+ free(buffer_);
54
+ }
55
+
56
+ uintptr_t position() const {
57
+ return position_;
58
+ }
59
+
60
+ template<typename T>
61
+ class Slot {
62
+ public:
63
+ Slot(Writer* w, uintptr_t offset) : w_(w), offset_(offset) { }
64
+
65
+ T* operator-> () {
66
+ return w_->RawSlotAt<T>(offset_);
67
+ }
68
+
69
+ void set(const T& value) {
70
+ *w_->RawSlotAt<T>(offset_) = value;
71
+ }
72
+
73
+ Slot<T> at(int i) {
74
+ return Slot<T>(w_, offset_ + sizeof(T) * i);
75
+ }
76
+
77
+ private:
78
+ Writer* w_;
79
+ uintptr_t offset_;
80
+ };
81
+
82
+ template<typename T>
83
+ void Write(const T& val) {
84
+ Ensure(position_ + sizeof(T));
85
+ *RawSlotAt<T>(position_) = val;
86
+ position_ += sizeof(T);
87
+ }
88
+
89
+ template<typename T>
90
+ Slot<T> SlotAt(uintptr_t offset) {
91
+ Ensure(offset + sizeof(T));
92
+ return Slot<T>(this, offset);
93
+ }
94
+
95
+ template<typename T>
96
+ Slot<T> CreateSlotHere() {
97
+ return CreateSlotsHere<T>(1);
98
+ }
99
+
100
+ template<typename T>
101
+ Slot<T> CreateSlotsHere(uint32_t count) {
102
+ uintptr_t slot_position = position_;
103
+ position_ += sizeof(T) * count;
104
+ Ensure(position_);
105
+ return SlotAt<T>(slot_position);
106
+ }
107
+
108
+ void Ensure(uintptr_t pos) {
109
+ if (capacity_ < pos) {
110
+ while (capacity_ < pos) capacity_ *= 2;
111
+ buffer_ = reinterpret_cast<byte*>(realloc(buffer_, capacity_));
112
+ }
113
+ }
114
+
115
+ ELF* elf() { return elf_; }
116
+
117
+ byte* buffer() { return buffer_; }
118
+
119
+ void Align(uintptr_t align) {
120
+ uintptr_t delta = position_ % align;
121
+ if (delta == 0) return;
122
+ uintptr_t padding = align - delta;
123
+ Ensure(position_ += padding);
124
+ ASSERT((position_ % align) == 0);
125
+ }
126
+
127
+ void WriteULEB128(uintptr_t value) {
128
+ do {
129
+ uint8_t byte = value & 0x7F;
130
+ value >>= 7;
131
+ if (value != 0) byte |= 0x80;
132
+ Write<uint8_t>(byte);
133
+ } while (value != 0);
134
+ }
135
+
136
+ void WriteSLEB128(intptr_t value) {
137
+ bool more = true;
138
+ while (more) {
139
+ int8_t byte = value & 0x7F;
140
+ bool byte_sign = byte & 0x40;
141
+ value >>= 7;
142
+
143
+ if ((value == 0 && !byte_sign) || (value == -1 && byte_sign)) {
144
+ more = false;
145
+ } else {
146
+ byte |= 0x80;
147
+ }
148
+
149
+ Write<int8_t>(byte);
150
+ }
151
+ }
152
+
153
+ void WriteString(const char* str) {
154
+ do {
155
+ Write<char>(*str);
156
+ } while (*str++);
157
+ }
158
+
159
+ private:
160
+ template<typename T> friend class Slot;
161
+
162
+ template<typename T>
163
+ T* RawSlotAt(uintptr_t offset) {
164
+ ASSERT(offset < capacity_ && offset + sizeof(T) <= capacity_);
165
+ return reinterpret_cast<T*>(&buffer_[offset]);
166
+ }
167
+
168
+ ELF* elf_;
169
+ uintptr_t position_;
170
+ uintptr_t capacity_;
171
+ byte* buffer_;
172
+ };
173
+
174
+ class StringTable;
175
+
176
+ class ELFSection : public ZoneObject {
177
+ public:
178
+ struct Header {
179
+ uint32_t name;
180
+ uint32_t type;
181
+ uintptr_t flags;
182
+ uintptr_t address;
183
+ uintptr_t offset;
184
+ uintptr_t size;
185
+ uint32_t link;
186
+ uint32_t info;
187
+ uintptr_t alignment;
188
+ uintptr_t entry_size;
189
+ };
190
+
191
+ enum Type {
192
+ TYPE_NULL = 0,
193
+ TYPE_PROGBITS = 1,
194
+ TYPE_SYMTAB = 2,
195
+ TYPE_STRTAB = 3,
196
+ TYPE_RELA = 4,
197
+ TYPE_HASH = 5,
198
+ TYPE_DYNAMIC = 6,
199
+ TYPE_NOTE = 7,
200
+ TYPE_NOBITS = 8,
201
+ TYPE_REL = 9,
202
+ TYPE_SHLIB = 10,
203
+ TYPE_DYNSYM = 11,
204
+ TYPE_LOPROC = 0x70000000,
205
+ TYPE_X86_64_UNWIND = 0x70000001,
206
+ TYPE_HIPROC = 0x7fffffff,
207
+ TYPE_LOUSER = 0x80000000,
208
+ TYPE_HIUSER = 0xffffffff
209
+ };
210
+
211
+ enum Flags {
212
+ FLAG_WRITE = 1,
213
+ FLAG_ALLOC = 2,
214
+ FLAG_EXEC = 4
215
+ };
216
+
217
+ enum SpecialIndexes {
218
+ INDEX_ABSOLUTE = 0xfff1
219
+ };
220
+
221
+ ELFSection(const char* name, Type type, uintptr_t align)
222
+ : name_(name), type_(type), align_(align) { }
223
+
224
+ virtual ~ELFSection() { }
225
+
226
+ void PopulateHeader(Writer::Slot<Header> header, StringTable* strtab);
227
+
228
+ virtual void WriteBody(Writer::Slot<Header> header, Writer* w) {
229
+ uintptr_t start = w->position();
230
+ if (WriteBody(w)) {
231
+ uintptr_t end = w->position();
232
+ header->offset = start;
233
+ header->size = end - start;
234
+ }
235
+ }
236
+
237
+ virtual bool WriteBody(Writer* w) {
238
+ return false;
239
+ }
240
+
241
+ uint16_t index() const { return index_; }
242
+ void set_index(uint16_t index) { index_ = index; }
243
+
244
+ protected:
245
+ virtual void PopulateHeader(Writer::Slot<Header> header) {
246
+ header->flags = 0;
247
+ header->address = 0;
248
+ header->offset = 0;
249
+ header->size = 0;
250
+ header->link = 0;
251
+ header->info = 0;
252
+ header->entry_size = 0;
253
+ }
254
+
255
+
256
+ private:
257
+ const char* name_;
258
+ Type type_;
259
+ uintptr_t align_;
260
+ uint16_t index_;
261
+ };
262
+
263
+
264
+ class FullHeaderELFSection : public ELFSection {
265
+ public:
266
+ FullHeaderELFSection(const char* name,
267
+ Type type,
268
+ uintptr_t align,
269
+ uintptr_t addr,
270
+ uintptr_t offset,
271
+ uintptr_t size,
272
+ uintptr_t flags)
273
+ : ELFSection(name, type, align),
274
+ addr_(addr),
275
+ offset_(offset),
276
+ size_(size),
277
+ flags_(flags) { }
278
+
279
+ protected:
280
+ virtual void PopulateHeader(Writer::Slot<Header> header) {
281
+ ELFSection::PopulateHeader(header);
282
+ header->address = addr_;
283
+ header->offset = offset_;
284
+ header->size = size_;
285
+ header->flags = flags_;
286
+ }
287
+
288
+ private:
289
+ uintptr_t addr_;
290
+ uintptr_t offset_;
291
+ uintptr_t size_;
292
+ uintptr_t flags_;
293
+ };
294
+
295
+
296
+ class StringTable : public ELFSection {
297
+ public:
298
+ explicit StringTable(const char* name)
299
+ : ELFSection(name, TYPE_STRTAB, 1), writer_(NULL), offset_(0), size_(0) {
300
+ }
301
+
302
+ uintptr_t Add(const char* str) {
303
+ if (*str == '\0') return 0;
304
+
305
+ uintptr_t offset = size_;
306
+ WriteString(str);
307
+ return offset;
308
+ }
309
+
310
+ void AttachWriter(Writer* w) {
311
+ writer_ = w;
312
+ offset_ = writer_->position();
313
+
314
+ // First entry in the string table should be an empty string.
315
+ WriteString("");
316
+ }
317
+
318
+ void DetachWriter() {
319
+ writer_ = NULL;
320
+ }
321
+
322
+ virtual void WriteBody(Writer::Slot<Header> header, Writer* w) {
323
+ ASSERT(writer_ == NULL);
324
+ header->offset = offset_;
325
+ header->size = size_;
326
+ }
327
+
328
+ private:
329
+ void WriteString(const char* str) {
330
+ uintptr_t written = 0;
331
+ do {
332
+ writer_->Write(*str);
333
+ written++;
334
+ } while (*str++);
335
+ size_ += written;
336
+ }
337
+
338
+ Writer* writer_;
339
+
340
+ uintptr_t offset_;
341
+ uintptr_t size_;
342
+ };
343
+
344
+
345
+ void ELFSection::PopulateHeader(Writer::Slot<ELFSection::Header> header,
346
+ StringTable* strtab) {
347
+ header->name = strtab->Add(name_);
348
+ header->type = type_;
349
+ header->alignment = align_;
350
+ PopulateHeader(header);
351
+ }
352
+
353
+
354
+ class ELF BASE_EMBEDDED {
355
+ public:
356
+ ELF() : sections_(6) {
357
+ sections_.Add(new ELFSection("", ELFSection::TYPE_NULL, 0));
358
+ sections_.Add(new StringTable(".shstrtab"));
359
+ }
360
+
361
+ void Write(Writer* w) {
362
+ WriteHeader(w);
363
+ WriteSectionTable(w);
364
+ WriteSections(w);
365
+ }
366
+
367
+ ELFSection* SectionAt(uint32_t index) {
368
+ return sections_[index];
369
+ }
370
+
371
+ uint32_t AddSection(ELFSection* section) {
372
+ sections_.Add(section);
373
+ section->set_index(sections_.length() - 1);
374
+ return sections_.length() - 1;
375
+ }
376
+
377
+ private:
378
+ struct ELFHeader {
379
+ uint8_t ident[16];
380
+ uint16_t type;
381
+ uint16_t machine;
382
+ uint32_t version;
383
+ uintptr_t entry;
384
+ uintptr_t pht_offset;
385
+ uintptr_t sht_offset;
386
+ uint32_t flags;
387
+ uint16_t header_size;
388
+ uint16_t pht_entry_size;
389
+ uint16_t pht_entry_num;
390
+ uint16_t sht_entry_size;
391
+ uint16_t sht_entry_num;
392
+ uint16_t sht_strtab_index;
393
+ };
394
+
395
+
396
+ void WriteHeader(Writer* w) {
397
+ ASSERT(w->position() == 0);
398
+ Writer::Slot<ELFHeader> header = w->CreateSlotHere<ELFHeader>();
399
+ #if defined(V8_TARGET_ARCH_IA32) || defined(V8_TARGET_ARCH_ARM)
400
+ const uint8_t ident[16] =
401
+ { 0x7f, 'E', 'L', 'F', 1, 1, 1, 0, 0, 0, 0, 0, 0, 0, 0, 0};
402
+ #elif defined(V8_TARGET_ARCH_X64)
403
+ const uint8_t ident[16] =
404
+ { 0x7f, 'E', 'L', 'F', 2, 1, 1, 0, 0, 0 , 0, 0, 0, 0, 0, 0};
405
+ #else
406
+ #error Unsupported target architecture.
407
+ #endif
408
+ memcpy(header->ident, ident, 16);
409
+ header->type = 1;
410
+ #if defined(V8_TARGET_ARCH_IA32)
411
+ header->machine = 3;
412
+ #elif defined(V8_TARGET_ARCH_X64)
413
+ // Processor identification value for x64 is 62 as defined in
414
+ // System V ABI, AMD64 Supplement
415
+ // http://www.x86-64.org/documentation/abi.pdf
416
+ header->machine = 62;
417
+ #elif defined(V8_TARGET_ARCH_ARM)
418
+ // Set to EM_ARM, defined as 40, in "ARM ELF File Format" at
419
+ // infocenter.arm.com/help/topic/com.arm.doc.dui0101a/DUI0101A_Elf.pdf
420
+ header->machine = 40;
421
+ #else
422
+ #error Unsupported target architecture.
423
+ #endif
424
+ header->version = 1;
425
+ header->entry = 0;
426
+ header->pht_offset = 0;
427
+ header->sht_offset = sizeof(ELFHeader); // Section table follows header.
428
+ header->flags = 0;
429
+ header->header_size = sizeof(ELFHeader);
430
+ header->pht_entry_size = 0;
431
+ header->pht_entry_num = 0;
432
+ header->sht_entry_size = sizeof(ELFSection::Header);
433
+ header->sht_entry_num = sections_.length();
434
+ header->sht_strtab_index = 1;
435
+ }
436
+
437
+ void WriteSectionTable(Writer* w) {
438
+ // Section headers table immediately follows file header.
439
+ ASSERT(w->position() == sizeof(ELFHeader));
440
+
441
+ Writer::Slot<ELFSection::Header> headers =
442
+ w->CreateSlotsHere<ELFSection::Header>(sections_.length());
443
+
444
+ // String table for section table is the first section.
445
+ StringTable* strtab = static_cast<StringTable*>(SectionAt(1));
446
+ strtab->AttachWriter(w);
447
+ for (int i = 0, length = sections_.length();
448
+ i < length;
449
+ i++) {
450
+ sections_[i]->PopulateHeader(headers.at(i), strtab);
451
+ }
452
+ strtab->DetachWriter();
453
+ }
454
+
455
+ int SectionHeaderPosition(uint32_t section_index) {
456
+ return sizeof(ELFHeader) + sizeof(ELFSection::Header) * section_index;
457
+ }
458
+
459
+ void WriteSections(Writer* w) {
460
+ Writer::Slot<ELFSection::Header> headers =
461
+ w->SlotAt<ELFSection::Header>(sizeof(ELFHeader));
462
+
463
+ for (int i = 0, length = sections_.length();
464
+ i < length;
465
+ i++) {
466
+ sections_[i]->WriteBody(headers.at(i), w);
467
+ }
468
+ }
469
+
470
+ ZoneList<ELFSection*> sections_;
471
+ };
472
+
473
+
474
+ class ELFSymbol BASE_EMBEDDED {
475
+ public:
476
+ enum Type {
477
+ TYPE_NOTYPE = 0,
478
+ TYPE_OBJECT = 1,
479
+ TYPE_FUNC = 2,
480
+ TYPE_SECTION = 3,
481
+ TYPE_FILE = 4,
482
+ TYPE_LOPROC = 13,
483
+ TYPE_HIPROC = 15
484
+ };
485
+
486
+ enum Binding {
487
+ BIND_LOCAL = 0,
488
+ BIND_GLOBAL = 1,
489
+ BIND_WEAK = 2,
490
+ BIND_LOPROC = 13,
491
+ BIND_HIPROC = 15
492
+ };
493
+
494
+ ELFSymbol(const char* name,
495
+ uintptr_t value,
496
+ uintptr_t size,
497
+ Binding binding,
498
+ Type type,
499
+ uint16_t section)
500
+ : name(name),
501
+ value(value),
502
+ size(size),
503
+ info((binding << 4) | type),
504
+ other(0),
505
+ section(section) {
506
+ }
507
+
508
+ Binding binding() const {
509
+ return static_cast<Binding>(info >> 4);
510
+ }
511
+ #if defined(V8_TARGET_ARCH_IA32) || defined(V8_TARGET_ARCH_ARM)
512
+ struct SerializedLayout {
513
+ SerializedLayout(uint32_t name,
514
+ uintptr_t value,
515
+ uintptr_t size,
516
+ Binding binding,
517
+ Type type,
518
+ uint16_t section)
519
+ : name(name),
520
+ value(value),
521
+ size(size),
522
+ info((binding << 4) | type),
523
+ other(0),
524
+ section(section) {
525
+ }
526
+
527
+ uint32_t name;
528
+ uintptr_t value;
529
+ uintptr_t size;
530
+ uint8_t info;
531
+ uint8_t other;
532
+ uint16_t section;
533
+ };
534
+ #elif defined(V8_TARGET_ARCH_X64)
535
+ struct SerializedLayout {
536
+ SerializedLayout(uint32_t name,
537
+ uintptr_t value,
538
+ uintptr_t size,
539
+ Binding binding,
540
+ Type type,
541
+ uint16_t section)
542
+ : name(name),
543
+ info((binding << 4) | type),
544
+ other(0),
545
+ section(section),
546
+ value(value),
547
+ size(size) {
548
+ }
549
+
550
+ uint32_t name;
551
+ uint8_t info;
552
+ uint8_t other;
553
+ uint16_t section;
554
+ uintptr_t value;
555
+ uintptr_t size;
556
+ };
557
+ #endif
558
+
559
+ void Write(Writer::Slot<SerializedLayout> s, StringTable* t) {
560
+ // Convert symbol names from strings to indexes in the string table.
561
+ s->name = t->Add(name);
562
+ s->value = value;
563
+ s->size = size;
564
+ s->info = info;
565
+ s->other = other;
566
+ s->section = section;
567
+ }
568
+
569
+ private:
570
+ const char* name;
571
+ uintptr_t value;
572
+ uintptr_t size;
573
+ uint8_t info;
574
+ uint8_t other;
575
+ uint16_t section;
576
+ };
577
+
578
+
579
+ class ELFSymbolTable : public ELFSection {
580
+ public:
581
+ explicit ELFSymbolTable(const char* name)
582
+ : ELFSection(name, TYPE_SYMTAB, sizeof(uintptr_t)),
583
+ locals_(1),
584
+ globals_(1) {
585
+ }
586
+
587
+ virtual void WriteBody(Writer::Slot<Header> header, Writer* w) {
588
+ w->Align(header->alignment);
589
+ int total_symbols = locals_.length() + globals_.length() + 1;
590
+ header->offset = w->position();
591
+
592
+ Writer::Slot<ELFSymbol::SerializedLayout> symbols =
593
+ w->CreateSlotsHere<ELFSymbol::SerializedLayout>(total_symbols);
594
+
595
+ header->size = w->position() - header->offset;
596
+
597
+ // String table for this symbol table should follow it in the section table.
598
+ StringTable* strtab =
599
+ static_cast<StringTable*>(w->elf()->SectionAt(index() + 1));
600
+ strtab->AttachWriter(w);
601
+ symbols.at(0).set(ELFSymbol::SerializedLayout(0,
602
+ 0,
603
+ 0,
604
+ ELFSymbol::BIND_LOCAL,
605
+ ELFSymbol::TYPE_NOTYPE,
606
+ 0));
607
+ WriteSymbolsList(&locals_, symbols.at(1), strtab);
608
+ WriteSymbolsList(&globals_, symbols.at(locals_.length() + 1), strtab);
609
+ strtab->DetachWriter();
610
+ }
611
+
612
+ void Add(const ELFSymbol& symbol) {
613
+ if (symbol.binding() == ELFSymbol::BIND_LOCAL) {
614
+ locals_.Add(symbol);
615
+ } else {
616
+ globals_.Add(symbol);
617
+ }
618
+ }
619
+
620
+ protected:
621
+ virtual void PopulateHeader(Writer::Slot<Header> header) {
622
+ ELFSection::PopulateHeader(header);
623
+ // We are assuming that string table will follow symbol table.
624
+ header->link = index() + 1;
625
+ header->info = locals_.length() + 1;
626
+ header->entry_size = sizeof(ELFSymbol::SerializedLayout);
627
+ }
628
+
629
+ private:
630
+ void WriteSymbolsList(const ZoneList<ELFSymbol>* src,
631
+ Writer::Slot<ELFSymbol::SerializedLayout> dst,
632
+ StringTable* strtab) {
633
+ for (int i = 0, len = src->length();
634
+ i < len;
635
+ i++) {
636
+ src->at(i).Write(dst.at(i), strtab);
637
+ }
638
+ }
639
+
640
+ ZoneList<ELFSymbol> locals_;
641
+ ZoneList<ELFSymbol> globals_;
642
+ };
643
+
644
+
645
+ class CodeDescription BASE_EMBEDDED {
646
+ public:
647
+
648
+ #ifdef V8_TARGET_ARCH_X64
649
+ enum StackState {
650
+ POST_RBP_PUSH,
651
+ POST_RBP_SET,
652
+ POST_RBP_POP,
653
+ STACK_STATE_MAX
654
+ };
655
+ #endif
656
+
657
+ CodeDescription(const char* name,
658
+ Code* code,
659
+ Handle<Script> script,
660
+ GDBJITLineInfo* lineinfo,
661
+ GDBJITInterface::CodeTag tag)
662
+ : name_(name),
663
+ code_(code),
664
+ script_(script),
665
+ lineinfo_(lineinfo),
666
+ tag_(tag) {
667
+ }
668
+
669
+ const char* name() const {
670
+ return name_;
671
+ }
672
+
673
+ GDBJITLineInfo* lineinfo() const {
674
+ return lineinfo_;
675
+ }
676
+
677
+ GDBJITInterface::CodeTag tag() const {
678
+ return tag_;
679
+ }
680
+
681
+ uintptr_t CodeStart() const {
682
+ return reinterpret_cast<uintptr_t>(code_->instruction_start());
683
+ }
684
+
685
+ uintptr_t CodeEnd() const {
686
+ return reinterpret_cast<uintptr_t>(code_->instruction_end());
687
+ }
688
+
689
+ uintptr_t CodeSize() const {
690
+ return CodeEnd() - CodeStart();
691
+ }
692
+
693
+ bool IsLineInfoAvailable() {
694
+ return !script_.is_null() &&
695
+ script_->source()->IsString() &&
696
+ script_->HasValidSource() &&
697
+ script_->name()->IsString() &&
698
+ lineinfo_ != NULL;
699
+ }
700
+
701
+ #ifdef V8_TARGET_ARCH_X64
702
+ uintptr_t GetStackStateStartAddress(StackState state) const {
703
+ ASSERT(state < STACK_STATE_MAX);
704
+ return stack_state_start_addresses_[state];
705
+ }
706
+
707
+ void SetStackStateStartAddress(StackState state, uintptr_t addr) {
708
+ ASSERT(state < STACK_STATE_MAX);
709
+ stack_state_start_addresses_[state] = addr;
710
+ }
711
+ #endif
712
+
713
+ SmartPointer<char> GetFilename() {
714
+ return String::cast(script_->name())->ToCString();
715
+ }
716
+
717
+ int GetScriptLineNumber(int pos) {
718
+ return GetScriptLineNumberSafe(script_, pos) + 1;
719
+ }
720
+
721
+
722
+ private:
723
+ const char* name_;
724
+ Code* code_;
725
+ Handle<Script> script_;
726
+ GDBJITLineInfo* lineinfo_;
727
+ GDBJITInterface::CodeTag tag_;
728
+ #ifdef V8_TARGET_ARCH_X64
729
+ uintptr_t stack_state_start_addresses_[STACK_STATE_MAX];
730
+ #endif
731
+ };
732
+
733
+
734
+ static void CreateSymbolsTable(CodeDescription* desc,
735
+ ELF* elf,
736
+ int text_section_index) {
737
+ ELFSymbolTable* symtab = new ELFSymbolTable(".symtab");
738
+ StringTable* strtab = new StringTable(".strtab");
739
+
740
+ // Symbol table should be followed by the linked string table.
741
+ elf->AddSection(symtab);
742
+ elf->AddSection(strtab);
743
+
744
+ symtab->Add(ELFSymbol("V8 Code",
745
+ 0,
746
+ 0,
747
+ ELFSymbol::BIND_LOCAL,
748
+ ELFSymbol::TYPE_FILE,
749
+ ELFSection::INDEX_ABSOLUTE));
750
+
751
+ symtab->Add(ELFSymbol(desc->name(),
752
+ 0,
753
+ desc->CodeSize(),
754
+ ELFSymbol::BIND_GLOBAL,
755
+ ELFSymbol::TYPE_FUNC,
756
+ text_section_index));
757
+ }
758
+
759
+
760
+ class DebugInfoSection : public ELFSection {
761
+ public:
762
+ explicit DebugInfoSection(CodeDescription* desc)
763
+ : ELFSection(".debug_info", TYPE_PROGBITS, 1), desc_(desc) { }
764
+
765
+ bool WriteBody(Writer* w) {
766
+ Writer::Slot<uint32_t> size = w->CreateSlotHere<uint32_t>();
767
+ uintptr_t start = w->position();
768
+ w->Write<uint16_t>(2); // DWARF version.
769
+ w->Write<uint32_t>(0); // Abbreviation table offset.
770
+ w->Write<uint8_t>(sizeof(intptr_t));
771
+
772
+ w->WriteULEB128(1); // Abbreviation code.
773
+ w->WriteString(*desc_->GetFilename());
774
+ w->Write<intptr_t>(desc_->CodeStart());
775
+ w->Write<intptr_t>(desc_->CodeStart() + desc_->CodeSize());
776
+ w->Write<uint32_t>(0);
777
+ size.set(static_cast<uint32_t>(w->position() - start));
778
+ return true;
779
+ }
780
+
781
+ private:
782
+ CodeDescription* desc_;
783
+ };
784
+
785
+
786
+ class DebugAbbrevSection : public ELFSection {
787
+ public:
788
+ DebugAbbrevSection() : ELFSection(".debug_abbrev", TYPE_PROGBITS, 1) { }
789
+
790
+ // DWARF2 standard, figure 14.
791
+ enum DWARF2Tags {
792
+ DW_TAG_COMPILE_UNIT = 0x11
793
+ };
794
+
795
+ // DWARF2 standard, figure 16.
796
+ enum DWARF2ChildrenDetermination {
797
+ DW_CHILDREN_NO = 0,
798
+ DW_CHILDREN_YES = 1
799
+ };
800
+
801
+ // DWARF standard, figure 17.
802
+ enum DWARF2Attribute {
803
+ DW_AT_NAME = 0x3,
804
+ DW_AT_STMT_LIST = 0x10,
805
+ DW_AT_LOW_PC = 0x11,
806
+ DW_AT_HIGH_PC = 0x12
807
+ };
808
+
809
+ // DWARF2 standard, figure 19.
810
+ enum DWARF2AttributeForm {
811
+ DW_FORM_ADDR = 0x1,
812
+ DW_FORM_STRING = 0x8,
813
+ DW_FORM_DATA4 = 0x6
814
+ };
815
+
816
+ bool WriteBody(Writer* w) {
817
+ w->WriteULEB128(1);
818
+ w->WriteULEB128(DW_TAG_COMPILE_UNIT);
819
+ w->Write<uint8_t>(DW_CHILDREN_NO);
820
+ w->WriteULEB128(DW_AT_NAME);
821
+ w->WriteULEB128(DW_FORM_STRING);
822
+ w->WriteULEB128(DW_AT_LOW_PC);
823
+ w->WriteULEB128(DW_FORM_ADDR);
824
+ w->WriteULEB128(DW_AT_HIGH_PC);
825
+ w->WriteULEB128(DW_FORM_ADDR);
826
+ w->WriteULEB128(DW_AT_STMT_LIST);
827
+ w->WriteULEB128(DW_FORM_DATA4);
828
+ w->WriteULEB128(0);
829
+ w->WriteULEB128(0);
830
+ w->WriteULEB128(0);
831
+ return true;
832
+ }
833
+ };
834
+
835
+
836
+ class DebugLineSection : public ELFSection {
837
+ public:
838
+ explicit DebugLineSection(CodeDescription* desc)
839
+ : ELFSection(".debug_line", TYPE_PROGBITS, 1),
840
+ desc_(desc) { }
841
+
842
+ // DWARF2 standard, figure 34.
843
+ enum DWARF2Opcodes {
844
+ DW_LNS_COPY = 1,
845
+ DW_LNS_ADVANCE_PC = 2,
846
+ DW_LNS_ADVANCE_LINE = 3,
847
+ DW_LNS_SET_FILE = 4,
848
+ DW_LNS_SET_COLUMN = 5,
849
+ DW_LNS_NEGATE_STMT = 6
850
+ };
851
+
852
+ // DWARF2 standard, figure 35.
853
+ enum DWARF2ExtendedOpcode {
854
+ DW_LNE_END_SEQUENCE = 1,
855
+ DW_LNE_SET_ADDRESS = 2,
856
+ DW_LNE_DEFINE_FILE = 3
857
+ };
858
+
859
+ bool WriteBody(Writer* w) {
860
+ // Write prologue.
861
+ Writer::Slot<uint32_t> total_length = w->CreateSlotHere<uint32_t>();
862
+ uintptr_t start = w->position();
863
+
864
+ // Used for special opcodes
865
+ const int8_t line_base = 1;
866
+ const uint8_t line_range = 7;
867
+ const int8_t max_line_incr = (line_base + line_range - 1);
868
+ const uint8_t opcode_base = DW_LNS_NEGATE_STMT + 1;
869
+
870
+ w->Write<uint16_t>(2); // Field version.
871
+ Writer::Slot<uint32_t> prologue_length = w->CreateSlotHere<uint32_t>();
872
+ uintptr_t prologue_start = w->position();
873
+ w->Write<uint8_t>(1); // Field minimum_instruction_length.
874
+ w->Write<uint8_t>(1); // Field default_is_stmt.
875
+ w->Write<int8_t>(line_base); // Field line_base.
876
+ w->Write<uint8_t>(line_range); // Field line_range.
877
+ w->Write<uint8_t>(opcode_base); // Field opcode_base.
878
+ w->Write<uint8_t>(0); // DW_LNS_COPY operands count.
879
+ w->Write<uint8_t>(1); // DW_LNS_ADVANCE_PC operands count.
880
+ w->Write<uint8_t>(1); // DW_LNS_ADVANCE_LINE operands count.
881
+ w->Write<uint8_t>(1); // DW_LNS_SET_FILE operands count.
882
+ w->Write<uint8_t>(1); // DW_LNS_SET_COLUMN operands count.
883
+ w->Write<uint8_t>(0); // DW_LNS_NEGATE_STMT operands count.
884
+ w->Write<uint8_t>(0); // Empty include_directories sequence.
885
+ w->WriteString(*desc_->GetFilename()); // File name.
886
+ w->WriteULEB128(0); // Current directory.
887
+ w->WriteULEB128(0); // Unknown modification time.
888
+ w->WriteULEB128(0); // Unknown file size.
889
+ w->Write<uint8_t>(0);
890
+ prologue_length.set(static_cast<uint32_t>(w->position() - prologue_start));
891
+
892
+ WriteExtendedOpcode(w, DW_LNE_SET_ADDRESS, sizeof(intptr_t));
893
+ w->Write<intptr_t>(desc_->CodeStart());
894
+ w->Write<uint8_t>(DW_LNS_COPY);
895
+
896
+ intptr_t pc = 0;
897
+ intptr_t line = 1;
898
+ bool is_statement = true;
899
+
900
+ List<GDBJITLineInfo::PCInfo>* pc_info = desc_->lineinfo()->pc_info();
901
+ pc_info->Sort(&ComparePCInfo);
902
+
903
+ int pc_info_length = pc_info->length();
904
+ for (int i = 0; i < pc_info_length; i++) {
905
+ GDBJITLineInfo::PCInfo* info = &pc_info->at(i);
906
+ ASSERT(info->pc_ >= pc);
907
+
908
+ // Reduce bloating in the debug line table by removing duplicate line
909
+ // entries (per DWARF2 standard).
910
+ intptr_t new_line = desc_->GetScriptLineNumber(info->pos_);
911
+ if (new_line == line) {
912
+ continue;
913
+ }
914
+
915
+ // Mark statement boundaries. For a better debugging experience, mark
916
+ // the last pc address in the function as a statement (e.g. "}"), so that
917
+ // a user can see the result of the last line executed in the function,
918
+ // should control reach the end.
919
+ if ((i+1) == pc_info_length) {
920
+ if (!is_statement) {
921
+ w->Write<uint8_t>(DW_LNS_NEGATE_STMT);
922
+ }
923
+ } else if (is_statement != info->is_statement_) {
924
+ w->Write<uint8_t>(DW_LNS_NEGATE_STMT);
925
+ is_statement = !is_statement;
926
+ }
927
+
928
+ // Generate special opcodes, if possible. This results in more compact
929
+ // debug line tables. See the DWARF 2.0 standard to learn more about
930
+ // special opcodes.
931
+ uintptr_t pc_diff = info->pc_ - pc;
932
+ intptr_t line_diff = new_line - line;
933
+
934
+ // Compute special opcode (see DWARF 2.0 standard)
935
+ intptr_t special_opcode = (line_diff - line_base) +
936
+ (line_range * pc_diff) + opcode_base;
937
+
938
+ // If special_opcode is less than or equal to 255, it can be used as a
939
+ // special opcode. If line_diff is larger than the max line increment
940
+ // allowed for a special opcode, or if line_diff is less than the minimum
941
+ // line that can be added to the line register (i.e. line_base), then
942
+ // special_opcode can't be used.
943
+ if ((special_opcode >= opcode_base) && (special_opcode <= 255) &&
944
+ (line_diff <= max_line_incr) && (line_diff >= line_base)) {
945
+ w->Write<uint8_t>(special_opcode);
946
+ } else {
947
+ w->Write<uint8_t>(DW_LNS_ADVANCE_PC);
948
+ w->WriteSLEB128(pc_diff);
949
+ w->Write<uint8_t>(DW_LNS_ADVANCE_LINE);
950
+ w->WriteSLEB128(line_diff);
951
+ w->Write<uint8_t>(DW_LNS_COPY);
952
+ }
953
+
954
+ // Increment the pc and line operands.
955
+ pc += pc_diff;
956
+ line += line_diff;
957
+ }
958
+ // Advance the pc to the end of the routine, since the end sequence opcode
959
+ // requires this.
960
+ w->Write<uint8_t>(DW_LNS_ADVANCE_PC);
961
+ w->WriteSLEB128(desc_->CodeSize() - pc);
962
+ WriteExtendedOpcode(w, DW_LNE_END_SEQUENCE, 0);
963
+ total_length.set(static_cast<uint32_t>(w->position() - start));
964
+ return true;
965
+ }
966
+
967
+ private:
968
+ void WriteExtendedOpcode(Writer* w,
969
+ DWARF2ExtendedOpcode op,
970
+ size_t operands_size) {
971
+ w->Write<uint8_t>(0);
972
+ w->WriteULEB128(operands_size + 1);
973
+ w->Write<uint8_t>(op);
974
+ }
975
+
976
+ static int ComparePCInfo(const GDBJITLineInfo::PCInfo* a,
977
+ const GDBJITLineInfo::PCInfo* b) {
978
+ if (a->pc_ == b->pc_) {
979
+ if (a->is_statement_ != b->is_statement_) {
980
+ return b->is_statement_ ? +1 : -1;
981
+ }
982
+ return 0;
983
+ } else if (a->pc_ > b->pc_) {
984
+ return +1;
985
+ } else {
986
+ return -1;
987
+ }
988
+ }
989
+
990
+ CodeDescription* desc_;
991
+ };
992
+
993
+
994
+ #ifdef V8_TARGET_ARCH_X64
995
+
996
+
997
+ class UnwindInfoSection : public ELFSection {
998
+ public:
999
+ explicit UnwindInfoSection(CodeDescription *desc);
1000
+ virtual bool WriteBody(Writer *w);
1001
+
1002
+ int WriteCIE(Writer *w);
1003
+ void WriteFDE(Writer *w, int);
1004
+
1005
+ void WriteFDEStateOnEntry(Writer *w);
1006
+ void WriteFDEStateAfterRBPPush(Writer *w);
1007
+ void WriteFDEStateAfterRBPSet(Writer *w);
1008
+ void WriteFDEStateAfterRBPPop(Writer *w);
1009
+
1010
+ void WriteLength(Writer *w,
1011
+ Writer::Slot<uint32_t>* length_slot,
1012
+ int initial_position);
1013
+
1014
+ private:
1015
+ CodeDescription *desc_;
1016
+
1017
+ // DWARF3 Specification, Table 7.23
1018
+ enum CFIInstructions {
1019
+ DW_CFA_ADVANCE_LOC = 0x40,
1020
+ DW_CFA_OFFSET = 0x80,
1021
+ DW_CFA_RESTORE = 0xC0,
1022
+ DW_CFA_NOP = 0x00,
1023
+ DW_CFA_SET_LOC = 0x01,
1024
+ DW_CFA_ADVANCE_LOC1 = 0x02,
1025
+ DW_CFA_ADVANCE_LOC2 = 0x03,
1026
+ DW_CFA_ADVANCE_LOC4 = 0x04,
1027
+ DW_CFA_OFFSET_EXTENDED = 0x05,
1028
+ DW_CFA_RESTORE_EXTENDED = 0x06,
1029
+ DW_CFA_UNDEFINED = 0x07,
1030
+ DW_CFA_SAME_VALUE = 0x08,
1031
+ DW_CFA_REGISTER = 0x09,
1032
+ DW_CFA_REMEMBER_STATE = 0x0A,
1033
+ DW_CFA_RESTORE_STATE = 0x0B,
1034
+ DW_CFA_DEF_CFA = 0x0C,
1035
+ DW_CFA_DEF_CFA_REGISTER = 0x0D,
1036
+ DW_CFA_DEF_CFA_OFFSET = 0x0E,
1037
+
1038
+ DW_CFA_DEF_CFA_EXPRESSION = 0x0F,
1039
+ DW_CFA_EXPRESSION = 0x10,
1040
+ DW_CFA_OFFSET_EXTENDED_SF = 0x11,
1041
+ DW_CFA_DEF_CFA_SF = 0x12,
1042
+ DW_CFA_DEF_CFA_OFFSET_SF = 0x13,
1043
+ DW_CFA_VAL_OFFSET = 0x14,
1044
+ DW_CFA_VAL_OFFSET_SF = 0x15,
1045
+ DW_CFA_VAL_EXPRESSION = 0x16
1046
+ };
1047
+
1048
+ // System V ABI, AMD64 Supplement, Version 0.99.5, Figure 3.36
1049
+ enum RegisterMapping {
1050
+ // Only the relevant ones have been added to reduce clutter.
1051
+ AMD64_RBP = 6,
1052
+ AMD64_RSP = 7,
1053
+ AMD64_RA = 16
1054
+ };
1055
+
1056
+ enum CFIConstants {
1057
+ CIE_ID = 0,
1058
+ CIE_VERSION = 1,
1059
+ CODE_ALIGN_FACTOR = 1,
1060
+ DATA_ALIGN_FACTOR = 1,
1061
+ RETURN_ADDRESS_REGISTER = AMD64_RA
1062
+ };
1063
+ };
1064
+
1065
+
1066
+ void UnwindInfoSection::WriteLength(Writer *w,
1067
+ Writer::Slot<uint32_t>* length_slot,
1068
+ int initial_position) {
1069
+ uint32_t align = (w->position() - initial_position) % kPointerSize;
1070
+
1071
+ if (align != 0) {
1072
+ for (uint32_t i = 0; i < (kPointerSize - align); i++) {
1073
+ w->Write<uint8_t>(DW_CFA_NOP);
1074
+ }
1075
+ }
1076
+
1077
+ ASSERT((w->position() - initial_position) % kPointerSize == 0);
1078
+ length_slot->set(w->position() - initial_position);
1079
+ }
1080
+
1081
+
1082
+ UnwindInfoSection::UnwindInfoSection(CodeDescription *desc)
1083
+ : ELFSection(".eh_frame", TYPE_X86_64_UNWIND, 1), desc_(desc)
1084
+ { }
1085
+
1086
+ int UnwindInfoSection::WriteCIE(Writer *w) {
1087
+ Writer::Slot<uint32_t> cie_length_slot = w->CreateSlotHere<uint32_t>();
1088
+ uint32_t cie_position = w->position();
1089
+
1090
+ // Write out the CIE header. Currently no 'common instructions' are
1091
+ // emitted onto the CIE; every FDE has its own set of instructions.
1092
+
1093
+ w->Write<uint32_t>(CIE_ID);
1094
+ w->Write<uint8_t>(CIE_VERSION);
1095
+ w->Write<uint8_t>(0); // Null augmentation string.
1096
+ w->WriteSLEB128(CODE_ALIGN_FACTOR);
1097
+ w->WriteSLEB128(DATA_ALIGN_FACTOR);
1098
+ w->Write<uint8_t>(RETURN_ADDRESS_REGISTER);
1099
+
1100
+ WriteLength(w, &cie_length_slot, cie_position);
1101
+
1102
+ return cie_position;
1103
+ }
1104
+
1105
+
1106
+ void UnwindInfoSection::WriteFDE(Writer *w, int cie_position) {
1107
+ // The only FDE for this function. The CFA is the current RBP.
1108
+ Writer::Slot<uint32_t> fde_length_slot = w->CreateSlotHere<uint32_t>();
1109
+ int fde_position = w->position();
1110
+ w->Write<int32_t>(fde_position - cie_position + 4);
1111
+
1112
+ w->Write<uintptr_t>(desc_->CodeStart());
1113
+ w->Write<uintptr_t>(desc_->CodeSize());
1114
+
1115
+ WriteFDEStateOnEntry(w);
1116
+ WriteFDEStateAfterRBPPush(w);
1117
+ WriteFDEStateAfterRBPSet(w);
1118
+ WriteFDEStateAfterRBPPop(w);
1119
+
1120
+ WriteLength(w, &fde_length_slot, fde_position);
1121
+ }
1122
+
1123
+
1124
+ void UnwindInfoSection::WriteFDEStateOnEntry(Writer *w) {
1125
+ // The first state, just after the control has been transferred to the the
1126
+ // function.
1127
+
1128
+ // RBP for this function will be the value of RSP after pushing the RBP
1129
+ // for the previous function. The previous RBP has not been pushed yet.
1130
+ w->Write<uint8_t>(DW_CFA_DEF_CFA_SF);
1131
+ w->WriteULEB128(AMD64_RSP);
1132
+ w->WriteSLEB128(-kPointerSize);
1133
+
1134
+ // The RA is stored at location CFA + kCallerPCOffset. This is an invariant,
1135
+ // and hence omitted from the next states.
1136
+ w->Write<uint8_t>(DW_CFA_OFFSET_EXTENDED);
1137
+ w->WriteULEB128(AMD64_RA);
1138
+ w->WriteSLEB128(StandardFrameConstants::kCallerPCOffset);
1139
+
1140
+ // The RBP of the previous function is still in RBP.
1141
+ w->Write<uint8_t>(DW_CFA_SAME_VALUE);
1142
+ w->WriteULEB128(AMD64_RBP);
1143
+
1144
+ // Last location described by this entry.
1145
+ w->Write<uint8_t>(DW_CFA_SET_LOC);
1146
+ w->Write<uint64_t>(
1147
+ desc_->GetStackStateStartAddress(CodeDescription::POST_RBP_PUSH));
1148
+ }
1149
+
1150
+
1151
+ void UnwindInfoSection::WriteFDEStateAfterRBPPush(Writer *w) {
1152
+ // The second state, just after RBP has been pushed.
1153
+
1154
+ // RBP / CFA for this function is now the current RSP, so just set the
1155
+ // offset from the previous rule (from -8) to 0.
1156
+ w->Write<uint8_t>(DW_CFA_DEF_CFA_OFFSET);
1157
+ w->WriteULEB128(0);
1158
+
1159
+ // The previous RBP is stored at CFA + kCallerFPOffset. This is an invariant
1160
+ // in this and the next state, and hence omitted in the next state.
1161
+ w->Write<uint8_t>(DW_CFA_OFFSET_EXTENDED);
1162
+ w->WriteULEB128(AMD64_RBP);
1163
+ w->WriteSLEB128(StandardFrameConstants::kCallerFPOffset);
1164
+
1165
+ // Last location described by this entry.
1166
+ w->Write<uint8_t>(DW_CFA_SET_LOC);
1167
+ w->Write<uint64_t>(
1168
+ desc_->GetStackStateStartAddress(CodeDescription::POST_RBP_SET));
1169
+ }
1170
+
1171
+
1172
+ void UnwindInfoSection::WriteFDEStateAfterRBPSet(Writer *w) {
1173
+ // The third state, after the RBP has been set.
1174
+
1175
+ // The CFA can now directly be set to RBP.
1176
+ w->Write<uint8_t>(DW_CFA_DEF_CFA);
1177
+ w->WriteULEB128(AMD64_RBP);
1178
+ w->WriteULEB128(0);
1179
+
1180
+ // Last location described by this entry.
1181
+ w->Write<uint8_t>(DW_CFA_SET_LOC);
1182
+ w->Write<uint64_t>(
1183
+ desc_->GetStackStateStartAddress(CodeDescription::POST_RBP_POP));
1184
+ }
1185
+
1186
+
1187
+ void UnwindInfoSection::WriteFDEStateAfterRBPPop(Writer *w) {
1188
+ // The fourth (final) state. The RBP has been popped (just before issuing a
1189
+ // return).
1190
+
1191
+ // The CFA can is now calculated in the same way as in the first state.
1192
+ w->Write<uint8_t>(DW_CFA_DEF_CFA_SF);
1193
+ w->WriteULEB128(AMD64_RSP);
1194
+ w->WriteSLEB128(-kPointerSize);
1195
+
1196
+ // The RBP
1197
+ w->Write<uint8_t>(DW_CFA_OFFSET_EXTENDED);
1198
+ w->WriteULEB128(AMD64_RBP);
1199
+ w->WriteSLEB128(StandardFrameConstants::kCallerFPOffset);
1200
+
1201
+ // Last location described by this entry.
1202
+ w->Write<uint8_t>(DW_CFA_SET_LOC);
1203
+ w->Write<uint64_t>(desc_->CodeEnd());
1204
+ }
1205
+
1206
+
1207
+ bool UnwindInfoSection::WriteBody(Writer *w) {
1208
+ uint32_t cie_position = WriteCIE(w);
1209
+ WriteFDE(w, cie_position);
1210
+ return true;
1211
+ }
1212
+
1213
+
1214
+ #endif // V8_TARGET_ARCH_X64
1215
+
1216
+
1217
+ static void CreateDWARFSections(CodeDescription* desc, ELF* elf) {
1218
+ if (desc->IsLineInfoAvailable()) {
1219
+ elf->AddSection(new DebugInfoSection(desc));
1220
+ elf->AddSection(new DebugAbbrevSection);
1221
+ elf->AddSection(new DebugLineSection(desc));
1222
+ }
1223
+ #ifdef V8_TARGET_ARCH_X64
1224
+ elf->AddSection(new UnwindInfoSection(desc));
1225
+ #endif
1226
+ }
1227
+
1228
+
1229
+ // -------------------------------------------------------------------
1230
+ // Binary GDB JIT Interface as described in
1231
+ // http://sourceware.org/gdb/onlinedocs/gdb/Declarations.html
1232
+ extern "C" {
1233
+ typedef enum {
1234
+ JIT_NOACTION = 0,
1235
+ JIT_REGISTER_FN,
1236
+ JIT_UNREGISTER_FN
1237
+ } JITAction;
1238
+
1239
+ struct JITCodeEntry {
1240
+ JITCodeEntry* next_;
1241
+ JITCodeEntry* prev_;
1242
+ Address symfile_addr_;
1243
+ uint64_t symfile_size_;
1244
+ };
1245
+
1246
+ struct JITDescriptor {
1247
+ uint32_t version_;
1248
+ uint32_t action_flag_;
1249
+ JITCodeEntry *relevant_entry_;
1250
+ JITCodeEntry *first_entry_;
1251
+ };
1252
+
1253
+ // GDB will place breakpoint into this function.
1254
+ // To prevent GCC from inlining or removing it we place noinline attribute
1255
+ // and inline assembler statement inside.
1256
+ void __attribute__((noinline)) __jit_debug_register_code() {
1257
+ __asm__("");
1258
+ }
1259
+
1260
+ // GDB will inspect contents of this descriptor.
1261
+ // Static initialization is necessary to prevent GDB from seeing
1262
+ // uninitialized descriptor.
1263
+ JITDescriptor __jit_debug_descriptor = { 1, 0, 0, 0 };
1264
+ }
1265
+
1266
+
1267
+ static JITCodeEntry* CreateCodeEntry(Address symfile_addr,
1268
+ uintptr_t symfile_size) {
1269
+ JITCodeEntry* entry = static_cast<JITCodeEntry*>(
1270
+ malloc(sizeof(JITCodeEntry) + symfile_size));
1271
+
1272
+ entry->symfile_addr_ = reinterpret_cast<Address>(entry + 1);
1273
+ entry->symfile_size_ = symfile_size;
1274
+ memcpy(entry->symfile_addr_, symfile_addr, symfile_size);
1275
+
1276
+ entry->prev_ = entry->next_ = NULL;
1277
+
1278
+ return entry;
1279
+ }
1280
+
1281
+
1282
+ static void DestroyCodeEntry(JITCodeEntry* entry) {
1283
+ free(entry);
1284
+ }
1285
+
1286
+
1287
+ static void RegisterCodeEntry(JITCodeEntry* entry) {
1288
+ #if defined(DEBUG) && !defined(WIN32)
1289
+ static int file_num = 0;
1290
+ if (FLAG_gdbjit_dump) {
1291
+ static const int kMaxFileNameSize = 64;
1292
+ static const char* kElfFilePrefix = "/tmp/elfdump";
1293
+ static const char* kObjFileExt = ".o";
1294
+ char file_name[64];
1295
+
1296
+ OS::SNPrintF(Vector<char>(file_name, kMaxFileNameSize), "%s%d%s",
1297
+ kElfFilePrefix, file_num++, kObjFileExt);
1298
+ WriteBytes(file_name, entry->symfile_addr_, entry->symfile_size_);
1299
+ }
1300
+ #endif
1301
+
1302
+ entry->next_ = __jit_debug_descriptor.first_entry_;
1303
+ if (entry->next_ != NULL) entry->next_->prev_ = entry;
1304
+ __jit_debug_descriptor.first_entry_ =
1305
+ __jit_debug_descriptor.relevant_entry_ = entry;
1306
+
1307
+ __jit_debug_descriptor.action_flag_ = JIT_REGISTER_FN;
1308
+ __jit_debug_register_code();
1309
+ }
1310
+
1311
+
1312
+ static void UnregisterCodeEntry(JITCodeEntry* entry) {
1313
+ if (entry->prev_ != NULL) {
1314
+ entry->prev_->next_ = entry->next_;
1315
+ } else {
1316
+ __jit_debug_descriptor.first_entry_ = entry->next_;
1317
+ }
1318
+
1319
+ if (entry->next_ != NULL) {
1320
+ entry->next_->prev_ = entry->prev_;
1321
+ }
1322
+
1323
+ __jit_debug_descriptor.relevant_entry_ = entry;
1324
+ __jit_debug_descriptor.action_flag_ = JIT_UNREGISTER_FN;
1325
+ __jit_debug_register_code();
1326
+ }
1327
+
1328
+
1329
+ static JITCodeEntry* CreateELFObject(CodeDescription* desc) {
1330
+ ZoneScope zone_scope(DELETE_ON_EXIT);
1331
+
1332
+ ELF elf;
1333
+ Writer w(&elf);
1334
+
1335
+ int text_section_index = elf.AddSection(
1336
+ new FullHeaderELFSection(".text",
1337
+ ELFSection::TYPE_NOBITS,
1338
+ kCodeAlignment,
1339
+ desc->CodeStart(),
1340
+ 0,
1341
+ desc->CodeSize(),
1342
+ ELFSection::FLAG_ALLOC | ELFSection::FLAG_EXEC));
1343
+
1344
+ CreateSymbolsTable(desc, &elf, text_section_index);
1345
+
1346
+ CreateDWARFSections(desc, &elf);
1347
+
1348
+ elf.Write(&w);
1349
+
1350
+ return CreateCodeEntry(w.buffer(), w.position());
1351
+ }
1352
+
1353
+
1354
+ static bool SameCodeObjects(void* key1, void* key2) {
1355
+ return key1 == key2;
1356
+ }
1357
+
1358
+
1359
+ static HashMap* GetEntries() {
1360
+ static HashMap* entries = NULL;
1361
+ if (entries == NULL) {
1362
+ entries = new HashMap(&SameCodeObjects);
1363
+ }
1364
+ return entries;
1365
+ }
1366
+
1367
+
1368
+ static uint32_t HashForCodeObject(Code* code) {
1369
+ static const uintptr_t kGoldenRatio = 2654435761u;
1370
+ uintptr_t hash = reinterpret_cast<uintptr_t>(code->address());
1371
+ return static_cast<uint32_t>((hash >> kCodeAlignmentBits) * kGoldenRatio);
1372
+ }
1373
+
1374
+
1375
+ static const intptr_t kLineInfoTag = 0x1;
1376
+
1377
+
1378
+ static bool IsLineInfoTagged(void* ptr) {
1379
+ return 0 != (reinterpret_cast<intptr_t>(ptr) & kLineInfoTag);
1380
+ }
1381
+
1382
+
1383
+ static void* TagLineInfo(GDBJITLineInfo* ptr) {
1384
+ return reinterpret_cast<void*>(
1385
+ reinterpret_cast<intptr_t>(ptr) | kLineInfoTag);
1386
+ }
1387
+
1388
+
1389
+ static GDBJITLineInfo* UntagLineInfo(void* ptr) {
1390
+ return reinterpret_cast<GDBJITLineInfo*>(
1391
+ reinterpret_cast<intptr_t>(ptr) & ~kLineInfoTag);
1392
+ }
1393
+
1394
+
1395
+ void GDBJITInterface::AddCode(Handle<String> name,
1396
+ Handle<Script> script,
1397
+ Handle<Code> code) {
1398
+ if (!FLAG_gdbjit) return;
1399
+
1400
+ // Force initialization of line_ends array.
1401
+ GetScriptLineNumber(script, 0);
1402
+
1403
+ if (!name.is_null()) {
1404
+ SmartPointer<char> name_cstring = name->ToCString(DISALLOW_NULLS);
1405
+ AddCode(*name_cstring, *code, GDBJITInterface::FUNCTION, *script);
1406
+ } else {
1407
+ AddCode("", *code, GDBJITInterface::FUNCTION, *script);
1408
+ }
1409
+ }
1410
+
1411
+ static void AddUnwindInfo(CodeDescription *desc) {
1412
+ #ifdef V8_TARGET_ARCH_X64
1413
+ if (desc->tag() == GDBJITInterface::FUNCTION) {
1414
+ // To avoid propagating unwinding information through
1415
+ // compilation pipeline we use an approximation.
1416
+ // For most use cases this should not affect usability.
1417
+ static const int kFramePointerPushOffset = 1;
1418
+ static const int kFramePointerSetOffset = 4;
1419
+ static const int kFramePointerPopOffset = -3;
1420
+
1421
+ uintptr_t frame_pointer_push_address =
1422
+ desc->CodeStart() + kFramePointerPushOffset;
1423
+
1424
+ uintptr_t frame_pointer_set_address =
1425
+ desc->CodeStart() + kFramePointerSetOffset;
1426
+
1427
+ uintptr_t frame_pointer_pop_address =
1428
+ desc->CodeEnd() + kFramePointerPopOffset;
1429
+
1430
+ desc->SetStackStateStartAddress(CodeDescription::POST_RBP_PUSH,
1431
+ frame_pointer_push_address);
1432
+ desc->SetStackStateStartAddress(CodeDescription::POST_RBP_SET,
1433
+ frame_pointer_set_address);
1434
+ desc->SetStackStateStartAddress(CodeDescription::POST_RBP_POP,
1435
+ frame_pointer_pop_address);
1436
+ } else {
1437
+ desc->SetStackStateStartAddress(CodeDescription::POST_RBP_PUSH,
1438
+ desc->CodeStart());
1439
+ desc->SetStackStateStartAddress(CodeDescription::POST_RBP_SET,
1440
+ desc->CodeStart());
1441
+ desc->SetStackStateStartAddress(CodeDescription::POST_RBP_POP,
1442
+ desc->CodeEnd());
1443
+ }
1444
+ #endif // V8_TARGET_ARCH_X64
1445
+ }
1446
+
1447
+
1448
+ Mutex* GDBJITInterface::mutex_ = OS::CreateMutex();
1449
+
1450
+
1451
+ void GDBJITInterface::AddCode(const char* name,
1452
+ Code* code,
1453
+ GDBJITInterface::CodeTag tag,
1454
+ Script* script) {
1455
+ if (!FLAG_gdbjit) return;
1456
+
1457
+ ScopedLock lock(mutex_);
1458
+ AssertNoAllocation no_gc;
1459
+
1460
+ HashMap::Entry* e = GetEntries()->Lookup(code, HashForCodeObject(code), true);
1461
+ if (e->value != NULL && !IsLineInfoTagged(e->value)) return;
1462
+
1463
+ GDBJITLineInfo* lineinfo = UntagLineInfo(e->value);
1464
+ CodeDescription code_desc(name,
1465
+ code,
1466
+ script != NULL ? Handle<Script>(script)
1467
+ : Handle<Script>(),
1468
+ lineinfo,
1469
+ tag);
1470
+
1471
+ if (!FLAG_gdbjit_full && !code_desc.IsLineInfoAvailable()) {
1472
+ delete lineinfo;
1473
+ GetEntries()->Remove(code, HashForCodeObject(code));
1474
+ return;
1475
+ }
1476
+
1477
+ AddUnwindInfo(&code_desc);
1478
+ JITCodeEntry* entry = CreateELFObject(&code_desc);
1479
+ ASSERT(!IsLineInfoTagged(entry));
1480
+
1481
+ delete lineinfo;
1482
+ e->value = entry;
1483
+
1484
+ RegisterCodeEntry(entry);
1485
+ }
1486
+
1487
+
1488
+ void GDBJITInterface::AddCode(GDBJITInterface::CodeTag tag,
1489
+ const char* name,
1490
+ Code* code) {
1491
+ if (!FLAG_gdbjit) return;
1492
+
1493
+ EmbeddedVector<char, 256> buffer;
1494
+ StringBuilder builder(buffer.start(), buffer.length());
1495
+
1496
+ builder.AddString(Tag2String(tag));
1497
+ if ((name != NULL) && (*name != '\0')) {
1498
+ builder.AddString(": ");
1499
+ builder.AddString(name);
1500
+ } else {
1501
+ builder.AddFormatted(": code object %p", static_cast<void*>(code));
1502
+ }
1503
+
1504
+ AddCode(builder.Finalize(), code, tag);
1505
+ }
1506
+
1507
+
1508
+ void GDBJITInterface::AddCode(GDBJITInterface::CodeTag tag,
1509
+ String* name,
1510
+ Code* code) {
1511
+ if (!FLAG_gdbjit) return;
1512
+ AddCode(tag, name != NULL ? *name->ToCString(DISALLOW_NULLS) : NULL, code);
1513
+ }
1514
+
1515
+
1516
+ void GDBJITInterface::AddCode(GDBJITInterface::CodeTag tag, Code* code) {
1517
+ if (!FLAG_gdbjit) return;
1518
+
1519
+ AddCode(tag, "", code);
1520
+ }
1521
+
1522
+
1523
+ void GDBJITInterface::RemoveCode(Code* code) {
1524
+ if (!FLAG_gdbjit) return;
1525
+
1526
+ ScopedLock lock(mutex_);
1527
+ HashMap::Entry* e = GetEntries()->Lookup(code,
1528
+ HashForCodeObject(code),
1529
+ false);
1530
+ if (e == NULL) return;
1531
+
1532
+ if (IsLineInfoTagged(e->value)) {
1533
+ delete UntagLineInfo(e->value);
1534
+ } else {
1535
+ JITCodeEntry* entry = static_cast<JITCodeEntry*>(e->value);
1536
+ UnregisterCodeEntry(entry);
1537
+ DestroyCodeEntry(entry);
1538
+ }
1539
+ e->value = NULL;
1540
+ GetEntries()->Remove(code, HashForCodeObject(code));
1541
+ }
1542
+
1543
+
1544
+ void GDBJITInterface::RegisterDetailedLineInfo(Code* code,
1545
+ GDBJITLineInfo* line_info) {
1546
+ ScopedLock lock(mutex_);
1547
+ ASSERT(!IsLineInfoTagged(line_info));
1548
+ HashMap::Entry* e = GetEntries()->Lookup(code, HashForCodeObject(code), true);
1549
+ ASSERT(e->value == NULL);
1550
+ e->value = TagLineInfo(line_info);
1551
+ }
1552
+
1553
+
1554
+ } } // namespace v8::internal
1555
+ #endif