2013-04-16 20:47:10 +00:00
|
|
|
/* -*- Mode: C++; tab-width: 8; indent-tabs-mode: nil; c-basic-offset: 4 -*-
|
|
|
|
* vim: set ts=8 sts=4 et sw=4 tw=99:
|
|
|
|
* This Source Code Form is subject to the terms of the Mozilla Public
|
2012-05-21 11:12:37 +00:00
|
|
|
* License, v. 2.0. If a copy of the MPL was not distributed with this
|
|
|
|
* file, You can obtain one at http://mozilla.org/MPL/2.0/. */
|
1998-11-05 08:57:24 +00:00
|
|
|
|
|
|
|
/*
|
|
|
|
* JavaScript Debugging support - Stepping support
|
|
|
|
*/
|
|
|
|
|
|
|
|
#include "jsd.h"
|
|
|
|
|
|
|
|
/*
|
|
|
|
* #define JSD_TRACE 1
|
|
|
|
*/
|
|
|
|
|
|
|
|
#ifdef JSD_TRACE
|
|
|
|
|
|
|
|
static char*
|
|
|
|
_indentSpaces(int i)
|
|
|
|
{
|
|
|
|
#define MAX_INDENT 63
|
2013-09-19 19:26:36 +00:00
|
|
|
static char* p = nullptr;
|
1998-11-05 08:57:24 +00:00
|
|
|
if(!p)
|
|
|
|
{
|
|
|
|
p = calloc(1, MAX_INDENT+1);
|
|
|
|
if(!p) return "";
|
|
|
|
memset(p, ' ', MAX_INDENT);
|
|
|
|
}
|
|
|
|
if(i > MAX_INDENT) return p;
|
|
|
|
return p + MAX_INDENT-i;
|
|
|
|
}
|
|
|
|
|
|
|
|
static void
|
2013-01-25 08:49:29 +00:00
|
|
|
_interpreterTrace(JSDContext* jsdc, JSContext *cx, JSAbstractFramePtr frame,
|
2013-08-08 22:53:04 +00:00
|
|
|
bool isConstructing, bool before)
|
1998-11-05 08:57:24 +00:00
|
|
|
{
|
2013-09-19 19:26:36 +00:00
|
|
|
JSDScript* jsdscript = nullptr;
|
1998-11-05 08:57:24 +00:00
|
|
|
JSScript * script;
|
|
|
|
static indent = 0;
|
2013-09-19 19:26:36 +00:00
|
|
|
JSString* funName = nullptr;
|
1998-11-05 08:57:24 +00:00
|
|
|
|
2013-01-25 08:49:29 +00:00
|
|
|
script = frame.script();
|
1998-11-05 08:57:24 +00:00
|
|
|
if(script)
|
|
|
|
{
|
|
|
|
JSD_LOCK_SCRIPTS(jsdc);
|
2013-01-25 08:49:29 +00:00
|
|
|
jsdscript = jsd_FindOrCreateJSDScript(jsdc, cx, script, frame);
|
1998-11-05 08:57:24 +00:00
|
|
|
JSD_UNLOCK_SCRIPTS(jsdc);
|
|
|
|
if(jsdscript)
|
2011-01-29 21:40:29 +00:00
|
|
|
funName = JSD_GetScriptFunctionId(jsdc, jsdscript);
|
1998-11-05 08:57:24 +00:00
|
|
|
}
|
2010-11-24 16:56:43 +00:00
|
|
|
|
|
|
|
if(before)
|
|
|
|
printf("%sentering ", _indentSpaces(indent++));
|
|
|
|
else
|
|
|
|
printf("%sleaving ", _indentSpaces(--indent));
|
|
|
|
|
|
|
|
if (!funName)
|
|
|
|
printf("TOP_LEVEL");
|
|
|
|
else
|
|
|
|
JS_FileEscapedString(stdout, funName, 0);
|
1998-11-05 08:57:24 +00:00
|
|
|
|
|
|
|
if(before)
|
|
|
|
{
|
2010-10-12 18:50:03 +00:00
|
|
|
jsval thisVal;
|
|
|
|
|
2013-01-25 08:49:29 +00:00
|
|
|
printf("%s this: ", isConstructing ? "constructing":"");
|
2010-10-12 18:50:03 +00:00
|
|
|
|
2013-01-25 08:49:29 +00:00
|
|
|
if (JS_GetFrameThis(cx, frame, &thisVal))
|
2012-01-11 08:23:05 +00:00
|
|
|
printf("0x%0llx", (uintptr_t) thisVal);
|
2010-10-12 18:50:03 +00:00
|
|
|
else
|
|
|
|
puts("<unavailable>");
|
1998-11-05 08:57:24 +00:00
|
|
|
}
|
2010-11-24 16:56:43 +00:00
|
|
|
printf("\n");
|
1998-11-05 08:57:24 +00:00
|
|
|
JS_ASSERT(indent >= 0);
|
|
|
|
}
|
|
|
|
#endif
|
|
|
|
|
2013-08-08 22:53:04 +00:00
|
|
|
bool
|
2013-01-25 08:49:29 +00:00
|
|
|
_callHook(JSDContext *jsdc, JSContext *cx, JSAbstractFramePtr frame, bool isConstructing,
|
2013-08-08 22:53:04 +00:00
|
|
|
bool before, unsigned type, JSD_CallHookProc hook, void *hookData)
|
1998-11-05 08:57:24 +00:00
|
|
|
{
|
2001-09-01 18:03:53 +00:00
|
|
|
JSDScript* jsdscript;
|
|
|
|
JSScript* jsscript;
|
2013-08-08 22:53:04 +00:00
|
|
|
bool hookresult = true;
|
2001-09-01 18:03:53 +00:00
|
|
|
|
|
|
|
if (!jsdc || !jsdc->inited)
|
2013-08-07 06:59:54 +00:00
|
|
|
return false;
|
2003-08-14 22:49:09 +00:00
|
|
|
|
2010-07-24 02:33:49 +00:00
|
|
|
if (!hook && !(jsdc->flags & JSD_COLLECT_PROFILE_DATA))
|
2003-08-14 22:49:09 +00:00
|
|
|
{
|
2010-07-24 02:33:49 +00:00
|
|
|
/* no hook to call, no profile data needs to be collected,
|
|
|
|
* so there is nothing to do here.
|
2003-08-14 22:49:09 +00:00
|
|
|
*/
|
|
|
|
return hookresult;
|
|
|
|
}
|
2013-01-25 08:49:29 +00:00
|
|
|
|
|
|
|
if (before && isConstructing) {
|
2013-03-06 16:41:43 +00:00
|
|
|
JS::RootedValue newObj(cx);
|
2013-01-25 08:49:29 +00:00
|
|
|
if (!frame.getThisValue(cx, &newObj))
|
2013-08-07 06:59:54 +00:00
|
|
|
return false;
|
2013-01-25 08:49:29 +00:00
|
|
|
jsd_Constructing(jsdc, cx, JSVAL_TO_OBJECT(newObj), frame);
|
2010-10-12 18:50:03 +00:00
|
|
|
}
|
1998-11-05 08:57:24 +00:00
|
|
|
|
2013-01-25 08:49:29 +00:00
|
|
|
jsscript = frame.script();
|
2002-02-27 09:24:14 +00:00
|
|
|
if (jsscript)
|
2001-09-01 18:03:53 +00:00
|
|
|
{
|
2002-02-27 09:24:14 +00:00
|
|
|
JSD_LOCK_SCRIPTS(jsdc);
|
2013-01-25 08:49:29 +00:00
|
|
|
jsdscript = jsd_FindOrCreateJSDScript(jsdc, cx, jsscript, frame);
|
2002-02-27 09:24:14 +00:00
|
|
|
JSD_UNLOCK_SCRIPTS(jsdc);
|
|
|
|
|
|
|
|
if (jsdscript)
|
|
|
|
{
|
|
|
|
if (JSD_IS_PROFILE_ENABLED(jsdc, jsdscript))
|
|
|
|
{
|
|
|
|
JSDProfileData *pdata;
|
|
|
|
pdata = jsd_GetScriptProfileData (jsdc, jsdscript);
|
|
|
|
if (pdata)
|
|
|
|
{
|
|
|
|
if (before)
|
|
|
|
{
|
2011-06-09 08:13:03 +00:00
|
|
|
if (!pdata->lastCallStart)
|
2002-02-27 09:24:14 +00:00
|
|
|
{
|
Bug 708735 - Use <stdint.h> types in JSAPI and throughout SpiderMonkey. Continue to provide the {u,}int{8,16,32,64} and JS{Uint,Int}{8,16,32,64} integer types through a single header, however, for a simpler backout strategy -- and also to ease the transition for embedders. r=timeless on switching the jsd API to use the <stdint.h> types, r=luke, r=dmandelin
2011-12-09 03:54:10 +00:00
|
|
|
int64_t now;
|
2005-08-02 15:53:59 +00:00
|
|
|
JSDProfileData *callerpdata;
|
|
|
|
|
|
|
|
/* Get the time just the once, for consistency. */
|
|
|
|
now = JS_Now();
|
|
|
|
/* This contains a pointer to the profile data for
|
|
|
|
* the caller of this function. */
|
|
|
|
callerpdata = jsdc->callingFunctionPData;
|
|
|
|
if (callerpdata)
|
|
|
|
{
|
Bug 708735 - Use <stdint.h> types in JSAPI and throughout SpiderMonkey. Continue to provide the {u,}int{8,16,32,64} and JS{Uint,Int}{8,16,32,64} integer types through a single header, however, for a simpler backout strategy -- and also to ease the transition for embedders. r=timeless on switching the jsd API to use the <stdint.h> types, r=luke, r=dmandelin
2011-12-09 03:54:10 +00:00
|
|
|
int64_t ll_delta;
|
2005-08-02 15:53:59 +00:00
|
|
|
pdata->caller = callerpdata;
|
|
|
|
/* We need to 'stop' the timer for the caller.
|
|
|
|
* Use time since last return if appropriate. */
|
2011-06-09 08:13:03 +00:00
|
|
|
ll_delta = jsdc->lastReturnTime
|
|
|
|
? now - jsdc->lastReturnTime
|
|
|
|
: now - callerpdata->lastCallStart;
|
|
|
|
callerpdata->runningTime += ll_delta;
|
2005-08-02 15:53:59 +00:00
|
|
|
}
|
|
|
|
/* We're the new current function, and no return
|
|
|
|
* has happened yet. */
|
|
|
|
jsdc->callingFunctionPData = pdata;
|
2008-04-25 18:40:05 +00:00
|
|
|
jsdc->lastReturnTime = 0;
|
2005-08-02 15:53:59 +00:00
|
|
|
/* This function has no running time (just been
|
|
|
|
* called!), and we'll need the call start time. */
|
2008-04-25 18:40:05 +00:00
|
|
|
pdata->runningTime = 0;
|
2005-08-02 15:53:59 +00:00
|
|
|
pdata->lastCallStart = now;
|
2002-02-27 09:24:14 +00:00
|
|
|
} else {
|
|
|
|
if (++pdata->recurseDepth > pdata->maxRecurseDepth)
|
|
|
|
pdata->maxRecurseDepth = pdata->recurseDepth;
|
2005-08-02 15:53:59 +00:00
|
|
|
}
|
2002-02-27 09:24:14 +00:00
|
|
|
/* make sure we're called for the return too. */
|
2013-08-07 06:59:54 +00:00
|
|
|
hookresult = true;
|
2011-06-09 08:13:03 +00:00
|
|
|
} else if (!pdata->recurseDepth && pdata->lastCallStart) {
|
Bug 708735 - Use <stdint.h> types in JSAPI and throughout SpiderMonkey. Continue to provide the {u,}int{8,16,32,64} and JS{Uint,Int}{8,16,32,64} integer types through a single header, however, for a simpler backout strategy -- and also to ease the transition for embedders. r=timeless on switching the jsd API to use the <stdint.h> types, r=luke, r=dmandelin
2011-12-09 03:54:10 +00:00
|
|
|
int64_t now, ll_delta;
|
2012-02-24 22:19:52 +00:00
|
|
|
double delta;
|
2002-02-27 09:24:14 +00:00
|
|
|
now = JS_Now();
|
2011-06-09 08:13:03 +00:00
|
|
|
ll_delta = now - pdata->lastCallStart;
|
2011-06-21 22:07:02 +00:00
|
|
|
delta = ll_delta;
|
2002-02-27 09:24:14 +00:00
|
|
|
delta /= 1000.0;
|
|
|
|
pdata->totalExecutionTime += delta;
|
2005-08-02 15:53:59 +00:00
|
|
|
/* minExecutionTime starts as 0, so we need to overwrite
|
|
|
|
* it on the first call always. */
|
|
|
|
if ((0 == pdata->callCount) ||
|
2002-02-27 09:24:14 +00:00
|
|
|
delta < pdata->minExecutionTime)
|
|
|
|
{
|
|
|
|
pdata->minExecutionTime = delta;
|
|
|
|
}
|
|
|
|
if (delta > pdata->maxExecutionTime)
|
|
|
|
pdata->maxExecutionTime = delta;
|
2005-08-02 15:53:59 +00:00
|
|
|
|
|
|
|
/* If we last returned from a function (as opposed to
|
|
|
|
* having last entered this function), we need to inc.
|
|
|
|
* the running total by the time delta since the last
|
|
|
|
* return, and use the running total instead of the
|
|
|
|
* delta calculated above. */
|
2011-06-09 08:13:03 +00:00
|
|
|
if (jsdc->lastReturnTime)
|
2005-08-02 15:53:59 +00:00
|
|
|
{
|
2005-08-02 18:46:14 +00:00
|
|
|
/* Add last chunk to running time, and use total
|
|
|
|
* running time as 'delta'. */
|
2011-06-09 08:13:03 +00:00
|
|
|
ll_delta = now - jsdc->lastReturnTime;
|
|
|
|
pdata->runningTime += ll_delta;
|
2011-06-21 22:07:02 +00:00
|
|
|
delta = pdata->runningTime;
|
2005-08-02 15:53:59 +00:00
|
|
|
delta /= 1000.0;
|
|
|
|
}
|
|
|
|
|
|
|
|
pdata->totalOwnExecutionTime += delta;
|
|
|
|
/* See minExecutionTime comment above. */
|
|
|
|
if ((0 == pdata->callCount) ||
|
|
|
|
delta < pdata->minOwnExecutionTime)
|
|
|
|
{
|
|
|
|
pdata->minOwnExecutionTime = delta;
|
|
|
|
}
|
|
|
|
if (delta > pdata->maxOwnExecutionTime)
|
|
|
|
pdata->maxOwnExecutionTime = delta;
|
|
|
|
|
|
|
|
/* Current function is now our caller. */
|
|
|
|
jsdc->callingFunctionPData = pdata->caller;
|
|
|
|
/* No hanging pointers, please. */
|
2013-09-19 19:26:36 +00:00
|
|
|
pdata->caller = nullptr;
|
2005-08-02 15:53:59 +00:00
|
|
|
/* Mark the time we returned, and indicate this
|
|
|
|
* function is no longer running. */
|
|
|
|
jsdc->lastReturnTime = now;
|
2008-04-25 18:40:05 +00:00
|
|
|
pdata->lastCallStart = 0;
|
2002-02-27 09:24:14 +00:00
|
|
|
++pdata->callCount;
|
|
|
|
} else if (pdata->recurseDepth) {
|
|
|
|
--pdata->recurseDepth;
|
|
|
|
++pdata->callCount;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
if (hook)
|
|
|
|
jsd_CallCallHook (jsdc, cx, type, hook, hookData);
|
|
|
|
} else {
|
|
|
|
if (hook)
|
|
|
|
hookresult =
|
|
|
|
jsd_CallCallHook (jsdc, cx, type, hook, hookData);
|
|
|
|
else
|
2013-08-07 06:59:54 +00:00
|
|
|
hookresult = true;
|
2001-09-01 18:03:53 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
2002-02-27 09:24:14 +00:00
|
|
|
|
1998-11-05 08:57:24 +00:00
|
|
|
#ifdef JSD_TRACE
|
2013-01-25 08:49:29 +00:00
|
|
|
_interpreterTrace(jsdc, cx, frame, isConstructing, before);
|
2013-08-07 06:59:54 +00:00
|
|
|
return true;
|
1998-11-05 08:57:24 +00:00
|
|
|
#else
|
2001-09-01 18:03:53 +00:00
|
|
|
return hookresult;
|
1998-11-05 08:57:24 +00:00
|
|
|
#endif
|
|
|
|
|
|
|
|
}
|
2001-09-01 18:03:53 +00:00
|
|
|
|
2008-09-06 22:21:43 +00:00
|
|
|
void *
|
2013-01-25 08:49:29 +00:00
|
|
|
jsd_FunctionCallHook(JSContext *cx, JSAbstractFramePtr frame, bool isConstructing,
|
2013-08-08 22:53:04 +00:00
|
|
|
bool before, bool *ok, void *closure)
|
2001-09-01 18:03:53 +00:00
|
|
|
{
|
|
|
|
JSDContext* jsdc;
|
|
|
|
JSD_CallHookProc hook;
|
|
|
|
void* hookData;
|
|
|
|
|
|
|
|
jsdc = (JSDContext*) closure;
|
|
|
|
|
|
|
|
/* local in case jsdc->functionHook gets cleared on another thread */
|
|
|
|
JSD_LOCK();
|
|
|
|
hook = jsdc->functionHook;
|
|
|
|
hookData = jsdc->functionHookData;
|
|
|
|
JSD_UNLOCK();
|
|
|
|
|
2013-01-25 08:49:29 +00:00
|
|
|
if (_callHook (jsdc, cx, frame, isConstructing, before,
|
2001-09-01 18:03:53 +00:00
|
|
|
(before) ? JSD_HOOK_FUNCTION_CALL : JSD_HOOK_FUNCTION_RETURN,
|
|
|
|
hook, hookData))
|
|
|
|
{
|
|
|
|
return closure;
|
|
|
|
}
|
|
|
|
|
2013-09-19 19:26:36 +00:00
|
|
|
return nullptr;
|
2001-09-01 18:03:53 +00:00
|
|
|
}
|
|
|
|
|
2008-09-06 22:21:43 +00:00
|
|
|
void *
|
2013-01-25 08:49:29 +00:00
|
|
|
jsd_TopLevelCallHook(JSContext *cx, JSAbstractFramePtr frame, bool isConstructing,
|
2013-08-08 22:53:04 +00:00
|
|
|
bool before, bool *ok, void *closure)
|
2001-09-01 18:03:53 +00:00
|
|
|
{
|
|
|
|
JSDContext* jsdc;
|
|
|
|
JSD_CallHookProc hook;
|
|
|
|
void* hookData;
|
|
|
|
|
|
|
|
jsdc = (JSDContext*) closure;
|
|
|
|
|
|
|
|
/* local in case jsdc->toplevelHook gets cleared on another thread */
|
|
|
|
JSD_LOCK();
|
|
|
|
hook = jsdc->toplevelHook;
|
|
|
|
hookData = jsdc->toplevelHookData;
|
|
|
|
JSD_UNLOCK();
|
|
|
|
|
2013-01-25 08:49:29 +00:00
|
|
|
if (_callHook (jsdc, cx, frame, isConstructing, before,
|
2001-09-01 18:03:53 +00:00
|
|
|
(before) ? JSD_HOOK_TOPLEVEL_START : JSD_HOOK_TOPLEVEL_END,
|
|
|
|
hook, hookData))
|
|
|
|
{
|
|
|
|
return closure;
|
|
|
|
}
|
|
|
|
|
2013-09-19 19:26:36 +00:00
|
|
|
return nullptr;
|
2001-09-01 18:03:53 +00:00
|
|
|
|
|
|
|
}
|