DiagHelperMethodWrapper.cpp 14 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270
  1. //-------------------------------------------------------------------------------------------------------
  2. // Copyright (C) Microsoft. All rights reserved.
  3. // Licensed under the MIT license. See LICENSE.txt file in the project root for full license information.
  4. //-------------------------------------------------------------------------------------------------------
  5. #include "RuntimeDebugPch.h"
  6. #include "Language/JavascriptStackWalker.h"
  7. namespace Js
  8. {
  9. AutoRegisterIgnoreExceptionWrapper::AutoRegisterIgnoreExceptionWrapper(ThreadContext* threadContext) :
  10. m_threadContext(threadContext)
  11. {
  12. AssertMsg(!IsRegistered(threadContext), "BuiltInWrapper is already registered.");
  13. m_threadContext->GetDebugManager()->GetDebuggingFlags()->SetIsBuiltInWrapperPresent(true);
  14. }
  15. AutoRegisterIgnoreExceptionWrapper::~AutoRegisterIgnoreExceptionWrapper()
  16. {
  17. m_threadContext->GetDebugManager()->GetDebuggingFlags()->SetIsBuiltInWrapperPresent(false);
  18. }
  19. // static
  20. bool AutoRegisterIgnoreExceptionWrapper::IsRegistered(ThreadContext* threadContext)
  21. {
  22. return threadContext->GetDebugManager()->GetDebuggingFlags()->IsBuiltInWrapperPresent();
  23. }
  24. // These are wrappers for helpers that can throw non-OOM / non-SO exceptions.
  25. // Under debugger, if "continue after exception" is on, we catch the exception and bail out to next statement.
  26. // IMPORTANT note:
  27. // - we are taking advantage of stack alignment, that's why we can say all args have size not greater than sizeof(Var),
  28. // for args that have less size, stack will be aligned, and next arg will start from alignment position,
  29. // while we can take the value of current arg at current position and ignore remaining bytes used for alignment.
  30. // - all these wrappers expect that arguments are not float/double
  31. // (double takes 8 bytes != stack alignment on x86 and ARM, double and float use different registers (VFP) rather than Var on ARM).
  32. typedef Var (__stdcall *OrigHelperMethod0)();
  33. typedef Var (__stdcall *OrigHelperMethod1)(Var arg1);
  34. typedef Var (__stdcall *OrigHelperMethod2)(Var arg1, Var arg2);
  35. typedef Var (__stdcall *OrigHelperMethod3)(Var arg1, Var arg2, Var arg3);
  36. typedef Var (__stdcall *OrigHelperMethod4)(Var arg1, Var arg2, Var arg3, Var arg4);
  37. typedef Var (__stdcall *OrigHelperMethod5)(Var arg1, Var arg2, Var arg3, Var arg4, Var arg5);
  38. typedef Var (__stdcall *OrigHelperMethod6)(Var arg1, Var arg2, Var arg3, Var arg4, Var arg5, Var arg6);
  39. typedef Var (__stdcall *OrigHelperMethod7)(Var arg1, Var arg2, Var arg3, Var arg4, Var arg5, Var arg6, Var arg7);
  40. typedef Var (__stdcall *OrigHelperMethod8)(Var arg1, Var arg2, Var arg3, Var arg4, Var arg5, Var arg6, Var arg7, Var arg8);
  41. typedef Var (__stdcall *OrigHelperMethod9)(Var arg1, Var arg2, Var arg3, Var arg4, Var arg5, Var arg6, Var arg7, Var arg8, Var arg9);
  42. typedef Var (__stdcall *OrigHelperMethod10)(Var arg1, Var arg2, Var arg3, Var arg4, Var arg5, Var arg6, Var arg7, Var arg8, Var arg9, Var arg10);
  43. typedef Var (__stdcall *OrigHelperMethod11)(Var arg1, Var arg2, Var arg3, Var arg4, Var arg5, Var arg6, Var arg7, Var arg8, Var arg9, Var arg10, Var arg11);
  44. typedef Var (__stdcall *OrigHelperMethod12)(Var arg1, Var arg2, Var arg3, Var arg4, Var arg5, Var arg6, Var arg7, Var arg8, Var arg9, Var arg10, Var arg11, Var arg12);
  45. typedef Var (__stdcall *OrigHelperMethod13)(Var arg1, Var arg2, Var arg3, Var arg4, Var arg5, Var arg6, Var arg7, Var arg8, Var arg9, Var arg10, Var arg11, Var arg12, Var arg13);
  46. typedef Var (__stdcall *OrigHelperMethod14)(Var arg1, Var arg2, Var arg3, Var arg4, Var arg5, Var arg6, Var arg7, Var arg8, Var arg9, Var arg10, Var arg11, Var arg12, Var arg13, Var arg14);
  47. typedef Var (__stdcall *OrigHelperMethod15)(Var arg1, Var arg2, Var arg3, Var arg4, Var arg5, Var arg6, Var arg7, Var arg8, Var arg9, Var arg10, Var arg11, Var arg12, Var arg13, Var arg14, Var arg15);
  48. typedef Var (__stdcall *OrigHelperMethod16)(Var arg1, Var arg2, Var arg3, Var arg4, Var arg5, Var arg6, Var arg7, Var arg8, Var arg9, Var arg10, Var arg11, Var arg12, Var arg13, Var arg14, Var arg15, Var arg16);
  49. template<typename Fn>
  50. Var HelperMethodWrapper(ScriptContext* scriptContext, Fn fn)
  51. {
  52. if (AutoRegisterIgnoreExceptionWrapper::IsRegistered(scriptContext->GetThreadContext()))
  53. {
  54. return fn();
  55. }
  56. else
  57. {
  58. AutoRegisterIgnoreExceptionWrapper autoWrapper(scriptContext->GetThreadContext());
  59. return HelperOrLibraryMethodWrapper<false>(scriptContext, fn);
  60. }
  61. }
  62. Var HelperMethodWrapper0(ScriptContext* scriptContext, void* origHelperAddr)
  63. {
  64. Assert(origHelperAddr);
  65. return HelperMethodWrapper(scriptContext, [=] {
  66. return ((OrigHelperMethod0)origHelperAddr)();
  67. });
  68. }
  69. Var HelperMethodWrapper1(ScriptContext* scriptContext, void* origHelperAddr, Var arg1)
  70. {
  71. Assert(origHelperAddr);
  72. return HelperMethodWrapper(scriptContext, [=] {
  73. return ((OrigHelperMethod1)origHelperAddr)(arg1);
  74. });
  75. }
  76. Var HelperMethodWrapper2(ScriptContext* scriptContext, void* origHelperAddr, Var arg1, Var arg2)
  77. {
  78. Assert(origHelperAddr);
  79. return HelperMethodWrapper(scriptContext, [=] {
  80. return ((OrigHelperMethod2)origHelperAddr)(arg1, arg2);
  81. });
  82. }
  83. Var HelperMethodWrapper3(ScriptContext* scriptContext, void* origHelperAddr, Var arg1, Var arg2, Var arg3)
  84. {
  85. Assert(origHelperAddr);
  86. return HelperMethodWrapper(scriptContext, [=] {
  87. return ((OrigHelperMethod3)origHelperAddr)(arg1, arg2, arg3);
  88. });
  89. }
  90. Var HelperMethodWrapper4(ScriptContext* scriptContext, void* origHelperAddr, Var arg1, Var arg2, Var arg3, Var arg4)
  91. {
  92. Assert(origHelperAddr);
  93. return HelperMethodWrapper(scriptContext, [=] {
  94. return ((OrigHelperMethod4)origHelperAddr)(arg1, arg2, arg3, arg4);
  95. });
  96. }
  97. Var HelperMethodWrapper5(ScriptContext* scriptContext, void* origHelperAddr, Var arg1, Var arg2, Var arg3, Var arg4, Var arg5)
  98. {
  99. Assert(origHelperAddr);
  100. return HelperMethodWrapper(scriptContext, [=] {
  101. return ((OrigHelperMethod5)origHelperAddr)(arg1, arg2, arg3, arg4, arg5);
  102. });
  103. }
  104. Var HelperMethodWrapper6(ScriptContext* scriptContext, void* origHelperAddr, Var arg1, Var arg2, Var arg3, Var arg4, Var arg5, Var arg6)
  105. {
  106. Assert(origHelperAddr);
  107. return HelperMethodWrapper(scriptContext, [=] {
  108. return ((OrigHelperMethod6)origHelperAddr)(arg1, arg2, arg3, arg4, arg5, arg6);
  109. });
  110. }
  111. Var HelperMethodWrapper7(ScriptContext* scriptContext, void* origHelperAddr, Var arg1, Var arg2, Var arg3, Var arg4, Var arg5, Var arg6, Var arg7)
  112. {
  113. Assert(origHelperAddr);
  114. return HelperMethodWrapper(scriptContext, [=] {
  115. return ((OrigHelperMethod7)origHelperAddr)(arg1, arg2, arg3, arg4, arg5, arg6, arg7);
  116. });
  117. }
  118. Var HelperMethodWrapper8(ScriptContext* scriptContext, void* origHelperAddr, Var arg1, Var arg2, Var arg3, Var arg4, Var arg5, Var arg6, Var arg7, Var arg8)
  119. {
  120. Assert(origHelperAddr);
  121. return HelperMethodWrapper(scriptContext, [=] {
  122. return ((OrigHelperMethod8)origHelperAddr)(arg1, arg2, arg3, arg4, arg5, arg6, arg7, arg8);
  123. });
  124. }
  125. Var HelperMethodWrapper9(ScriptContext* scriptContext, void* origHelperAddr, Var arg1, Var arg2, Var arg3, Var arg4, Var arg5, Var arg6, Var arg7, Var arg8, Var arg9)
  126. {
  127. Assert(origHelperAddr);
  128. return HelperMethodWrapper(scriptContext, [=] {
  129. return ((OrigHelperMethod9)origHelperAddr)(arg1, arg2, arg3, arg4, arg5, arg6, arg7, arg8, arg9);
  130. });
  131. }
  132. Var HelperMethodWrapper10(ScriptContext* scriptContext, void* origHelperAddr, Var arg1, Var arg2, Var arg3, Var arg4, Var arg5, Var arg6, Var arg7, Var arg8, Var arg9, Var arg10)
  133. {
  134. Assert(origHelperAddr);
  135. return HelperMethodWrapper(scriptContext, [=] {
  136. return ((OrigHelperMethod10)origHelperAddr)(arg1, arg2, arg3, arg4, arg5, arg6, arg7, arg8, arg9, arg10);
  137. });
  138. }
  139. Var HelperMethodWrapper11(ScriptContext* scriptContext, void* origHelperAddr, Var arg1, Var arg2, Var arg3, Var arg4, Var arg5, Var arg6, Var arg7, Var arg8, Var arg9, Var arg10, Var arg11)
  140. {
  141. Assert(origHelperAddr);
  142. return HelperMethodWrapper(scriptContext, [=] {
  143. return ((OrigHelperMethod11)origHelperAddr)(arg1, arg2, arg3, arg4, arg5, arg6, arg7, arg8, arg9, arg10, arg11);
  144. });
  145. }
  146. Var HelperMethodWrapper12(ScriptContext* scriptContext, void* origHelperAddr, Var arg1, Var arg2, Var arg3, Var arg4, Var arg5, Var arg6, Var arg7, Var arg8, Var arg9, Var arg10, Var arg11, Var arg12)
  147. {
  148. Assert(origHelperAddr);
  149. return HelperMethodWrapper(scriptContext, [=] {
  150. return ((OrigHelperMethod12)origHelperAddr)(arg1, arg2, arg3, arg4, arg5, arg6, arg7, arg8, arg9, arg10, arg11, arg12);
  151. });
  152. }
  153. Var HelperMethodWrapper13(ScriptContext* scriptContext, void* origHelperAddr, Var arg1, Var arg2, Var arg3, Var arg4, Var arg5, Var arg6, Var arg7, Var arg8, Var arg9, Var arg10, Var arg11, Var arg12, Var arg13)
  154. {
  155. Assert(origHelperAddr);
  156. return HelperMethodWrapper(scriptContext, [=] {
  157. return ((OrigHelperMethod13)origHelperAddr)(arg1, arg2, arg3, arg4, arg5, arg6, arg7, arg8, arg9, arg10, arg11, arg12, arg13);
  158. });
  159. }
  160. Var HelperMethodWrapper14(ScriptContext* scriptContext, void* origHelperAddr, Var arg1, Var arg2, Var arg3, Var arg4, Var arg5, Var arg6, Var arg7, Var arg8, Var arg9, Var arg10, Var arg11, Var arg12, Var arg13, Var arg14)
  161. {
  162. Assert(origHelperAddr);
  163. return HelperMethodWrapper(scriptContext, [=] {
  164. return ((OrigHelperMethod14)origHelperAddr)(arg1, arg2, arg3, arg4, arg5, arg6, arg7, arg8, arg9, arg10, arg11, arg12, arg13, arg14);
  165. });
  166. }
  167. Var HelperMethodWrapper15(ScriptContext* scriptContext, void* origHelperAddr, Var arg1, Var arg2, Var arg3, Var arg4, Var arg5, Var arg6, Var arg7, Var arg8, Var arg9, Var arg10, Var arg11, Var arg12, Var arg13, Var arg14, Var arg15)
  168. {
  169. Assert(origHelperAddr);
  170. return HelperMethodWrapper(scriptContext, [=] {
  171. return ((OrigHelperMethod15)origHelperAddr)(arg1, arg2, arg3, arg4, arg5, arg6, arg7, arg8, arg9, arg10, arg11, arg12, arg13, arg14, arg15);
  172. });
  173. }
  174. Var HelperMethodWrapper16(ScriptContext* scriptContext, void* origHelperAddr, Var arg1, Var arg2, Var arg3, Var arg4, Var arg5, Var arg6, Var arg7, Var arg8, Var arg9, Var arg10, Var arg11, Var arg12, Var arg13, Var arg14, Var arg15, Var arg16)
  175. {
  176. Assert(origHelperAddr);
  177. return HelperMethodWrapper(scriptContext, [=] {
  178. return ((OrigHelperMethod16)origHelperAddr)(arg1, arg2, arg3, arg4, arg5, arg6, arg7, arg8, arg9, arg10, arg11, arg12, arg13, arg14, arg15, arg16);
  179. });
  180. }
  181. template <bool doCheckParentInterpreterFrame>
  182. void HandleHelperOrLibraryMethodWrapperException(ScriptContext * scriptContext, JavascriptExceptionObject * exceptionObject)
  183. {
  184. Assert(scriptContext);
  185. Assert(exceptionObject);
  186. // Note: there also could be plain OutOfMemoryException and StackOverflowException, no special handling for these.
  187. if (!exceptionObject->IsDebuggerSkip() ||
  188. exceptionObject == scriptContext->GetThreadContext()->GetPendingOOMErrorObject() ||
  189. exceptionObject == scriptContext->GetThreadContext()->GetPendingSOErrorObject())
  190. {
  191. JavascriptExceptionOperators::DoThrowCheckClone(exceptionObject, scriptContext);
  192. }
  193. if (doCheckParentInterpreterFrame)
  194. {
  195. // Note: JavascriptStackWalker is slow, but this is not hot path at all.
  196. // Note: we treat internal script code (such as Intl) as library code, thus
  197. // ignore isLibraryCode=true callers.
  198. bool isTopUserFrameNative;
  199. bool isTopUserFrameJavaScript = Js::JavascriptStackWalker::TryIsTopJavaScriptFrameNative(
  200. scriptContext, &isTopUserFrameNative, /* ignoreLibraryCode = */ true);
  201. AssertMsg(isTopUserFrameJavaScript, "How could we get non-javascript frame on exception?");
  202. if (isTopUserFrameJavaScript && !isTopUserFrameNative)
  203. {
  204. // If parent frame is interpreter frame, it already has try-catch around all calls,
  205. // so that we don't need any special handling here.
  206. JavascriptExceptionOperators::DoThrowCheckClone(exceptionObject, scriptContext);
  207. }
  208. }
  209. Assert(exceptionObject->IsDebuggerSkip());
  210. int nextStatementOffset;
  211. int offsetFromDebugger = exceptionObject->GetByteCodeOffsetAfterDebuggerSkip();
  212. if (offsetFromDebugger != DebuggingFlags::InvalidByteCodeOffset)
  213. {
  214. // The offset is already set for us by debugger (such as by set next statement).
  215. nextStatementOffset = offsetFromDebugger;
  216. }
  217. else
  218. {
  219. ByteCodeReader reader;
  220. reader.Create(exceptionObject->GetFunctionBody(), exceptionObject->GetByteCodeOffset());
  221. // Determine offset for next statement here.
  222. if (!scriptContext->GetDebugContext()->GetProbeContainer()->GetNextUserStatementOffsetForAdvance(
  223. exceptionObject->GetFunctionBody(), &reader, exceptionObject->GetByteCodeOffset(), &nextStatementOffset))
  224. {
  225. // Can't advance.
  226. JavascriptExceptionOperators::DoThrowCheckClone(exceptionObject, scriptContext);
  227. }
  228. }
  229. // Continue after exception.
  230. // Note: for this scenario InterpreterStackFrame::DebugProcess resets its state,
  231. // looks like we don't need to that because we start with brand new interpreter frame.
  232. // Indicate to bailout check that we should bail out for/into debugger and set the byte code offset to one of next statement.
  233. scriptContext->GetThreadContext()->GetDebugManager()->GetDebuggingFlags()->SetByteCodeOffsetAndFuncAfterIgnoreException(
  234. nextStatementOffset, exceptionObject->GetFunctionBody()->GetFunctionNumber());
  235. }
  236. template void HandleHelperOrLibraryMethodWrapperException<true>(ScriptContext * scriptContext, JavascriptExceptionObject * exceptionObject);
  237. template void HandleHelperOrLibraryMethodWrapperException<false> (ScriptContext * scriptContext, JavascriptExceptionObject * exceptionObject);
  238. } // namespace Js