2001-04-17 12:50:41 +04:00
|
|
|
/* -*- Mode: C++; tab-width: 4; indent-tabs-mode: nil; c-basic-offset: 4 -*-
|
|
|
|
*
|
|
|
|
* The contents of this file are subject to the Mozilla Public License
|
|
|
|
* Version 1.1 (the "License"); you may not use this file except in
|
|
|
|
* compliance with the License. You may obtain a copy of the License at
|
|
|
|
* http://www.mozilla.org/MPL/
|
|
|
|
*
|
|
|
|
* Software distributed under the License is distributed on an "AS IS" basis,
|
|
|
|
* WITHOUT WARRANTY OF ANY KIND, either express or implied. See the License
|
|
|
|
* for the specific language governing rights and limitations under the
|
|
|
|
* License.
|
|
|
|
*
|
|
|
|
* The Original Code is mozilla.org code
|
|
|
|
*
|
|
|
|
* The Initial Developer of the Original Code is
|
|
|
|
* Netscape Communications Corporation
|
|
|
|
* Portions created by Netscape are
|
|
|
|
* Copyright (C) 1998 Netscape Communications Corporation.
|
|
|
|
*
|
|
|
|
* Alternatively, the contents of this file may be used under the
|
|
|
|
* terms of the GNU Public License (the "GPL"), in which case the
|
|
|
|
* provisions of the GPL are applicable instead of those above.
|
|
|
|
* If you wish to allow use of your version of this file only
|
|
|
|
* under the terms of the GPL and not to allow others to use your
|
|
|
|
* version of this file under the MPL, indicate your decision by
|
|
|
|
* deleting the provisions above and replace them with the notice
|
|
|
|
* and other provisions required by the GPL. If you do not delete
|
|
|
|
* the provisions above, a recipient may use your version of this
|
|
|
|
* file under either the MPL or the GPL.
|
|
|
|
*
|
|
|
|
* Contributor(s):
|
2001-04-20 07:54:05 +04:00
|
|
|
* Robert Ginda, <rginda@netscape.com>
|
2001-04-17 12:50:41 +04:00
|
|
|
*
|
|
|
|
*/
|
|
|
|
|
|
|
|
#include "jsd_xpc.h"
|
2001-08-29 02:02:07 +04:00
|
|
|
#include "jscntxt.h"
|
2001-04-17 12:50:41 +04:00
|
|
|
|
|
|
|
#include "nsIXPConnect.h"
|
|
|
|
#include "nsIGenericFactory.h"
|
|
|
|
#include "nsIServiceManager.h"
|
2001-10-30 16:41:32 +03:00
|
|
|
#include "nsIScriptGlobalObject.h"
|
|
|
|
#include "nsIObserver.h"
|
|
|
|
#include "nsIObserverService.h"
|
|
|
|
#include "nsICategoryManager.h"
|
|
|
|
#include "nsIJSRuntimeService.h"
|
2001-09-06 01:29:59 +04:00
|
|
|
#include "nsIEventQueueService.h"
|
2001-04-21 11:21:27 +04:00
|
|
|
#include "nsMemory.h"
|
2001-04-17 12:50:41 +04:00
|
|
|
#include "jsdebug.h"
|
2001-09-29 12:28:41 +04:00
|
|
|
#include "nsReadableUtils.h"
|
2001-04-17 12:50:41 +04:00
|
|
|
|
2001-04-20 07:54:05 +04:00
|
|
|
/* XXX this stuff is used by NestEventLoop, a temporary hack to be refactored
|
|
|
|
* later */
|
|
|
|
#include "nsWidgetsCID.h"
|
|
|
|
#include "nsIAppShell.h"
|
|
|
|
#include "nsIJSContextStack.h"
|
|
|
|
|
2001-08-29 02:02:07 +04:00
|
|
|
#define CAUTIOUS_SCRIPTHOOK
|
|
|
|
|
2001-07-04 02:19:04 +04:00
|
|
|
#ifdef DEBUG_verbose
|
|
|
|
# define DEBUG_COUNT(name, count) \
|
|
|
|
{ if ((count % 10) == 0) printf (name ": %i\n", count); }
|
|
|
|
# define DEBUG_CREATE(name, count) {count++; DEBUG_COUNT ("+++++ "name,count)}
|
|
|
|
# define DEBUG_DESTROY(name, count) {count--; DEBUG_COUNT ("----- "name,count)}
|
|
|
|
#else
|
2001-07-05 13:06:24 +04:00
|
|
|
# define DEBUG_CREATE(name, count)
|
|
|
|
# define DEBUG_DESTROY(name, count)
|
2001-07-04 02:19:04 +04:00
|
|
|
#endif
|
|
|
|
|
|
|
|
#define ASSERT_VALID_CONTEXT { if (!mCx) return NS_ERROR_NOT_AVAILABLE; }
|
|
|
|
#define ASSERT_VALID_FRAME { if (!mValid) return NS_ERROR_NOT_AVAILABLE; }
|
|
|
|
#define ASSERT_VALID_PROPERTY { if (!mValid) return NS_ERROR_NOT_AVAILABLE; }
|
|
|
|
#define ASSERT_VALID_SCRIPT { if (!mValid) return NS_ERROR_NOT_AVAILABLE; }
|
|
|
|
#define ASSERT_VALID_VALUE { if (!mValid) return NS_ERROR_NOT_AVAILABLE; }
|
2001-05-02 01:33:43 +04:00
|
|
|
|
2001-05-21 10:00:12 +04:00
|
|
|
#define JSDSERVICE_CID \
|
|
|
|
{ /* f1299dc2-1dd1-11b2-a347-ee6b7660e048 */ \
|
|
|
|
0xf1299dc2, \
|
|
|
|
0x1dd1, \
|
|
|
|
0x11b2, \
|
|
|
|
{0xa3, 0x47, 0xee, 0x6b, 0x76, 0x60, 0xe0, 0x48} \
|
|
|
|
}
|
|
|
|
|
2001-10-30 16:41:32 +03:00
|
|
|
#define JSDASO_CID \
|
|
|
|
{ /* 2fd6b7f6-eb8c-4f32-ad26-113f2c02d0fe */ \
|
|
|
|
0x2fd6b7f6, \
|
|
|
|
0xeb8c, \
|
|
|
|
0x4f32, \
|
|
|
|
{0xad, 0x26, 0x11, 0x3f, 0x2c, 0x02, 0xd0, 0xfe} \
|
|
|
|
}
|
|
|
|
|
|
|
|
#define NS_CATMAN_CTRID "@mozilla.org/categorymanager;1"
|
|
|
|
#define NS_JSRT_CTRID "@mozilla.org/js/xpc/RuntimeService;1"
|
|
|
|
|
|
|
|
#define AUTOREG_CATEGORY "xpcom-autoregistration"
|
|
|
|
#define APPSTART_CATEGORY "app-startup"
|
|
|
|
#define JSD_STARTUP_ENTRY "JSDebugger Startup Observer"
|
|
|
|
|
2001-09-22 05:39:24 +04:00
|
|
|
JS_STATIC_DLL_CALLBACK (JSBool)
|
2001-04-26 01:54:59 +04:00
|
|
|
jsds_GCCallbackProc (JSContext *cx, JSGCStatus status);
|
2001-05-02 01:33:43 +04:00
|
|
|
|
2001-04-26 01:54:59 +04:00
|
|
|
/*******************************************************************************
|
|
|
|
* global vars
|
|
|
|
*******************************************************************************/
|
|
|
|
|
2001-04-20 07:54:05 +04:00
|
|
|
static NS_DEFINE_CID(kAppShellCID, NS_APPSHELL_CID);
|
2001-09-06 01:29:59 +04:00
|
|
|
static NS_DEFINE_CID(kEventQueueServiceCID, NS_EVENTQUEUESERVICE_CID);
|
2001-04-20 07:54:05 +04:00
|
|
|
|
2001-09-06 01:29:59 +04:00
|
|
|
const char jsdServiceCtrID[] = "@mozilla.org/js/jsd/debugger-service;1";
|
2001-10-30 16:41:32 +03:00
|
|
|
const char jsdASObserverCtrID[] = "@mozilla.org/js/jsd/app-start-observer;2";
|
2001-04-17 12:50:41 +04:00
|
|
|
|
2001-05-05 02:14:04 +04:00
|
|
|
#ifdef DEBUG_verbose
|
2001-07-04 02:19:04 +04:00
|
|
|
PRUint32 gScriptCount = 0;
|
|
|
|
PRUint32 gValueCount = 0;
|
|
|
|
PRUint32 gPropertyCount = 0;
|
2001-05-05 02:14:04 +04:00
|
|
|
#endif
|
2001-05-02 01:33:43 +04:00
|
|
|
|
2001-04-26 01:54:59 +04:00
|
|
|
static jsdService *gJsds = 0;
|
|
|
|
static JSGCCallback gLastGCProc = jsds_GCCallbackProc;
|
|
|
|
static JSGCStatus gGCStatus = JSGC_END;
|
2001-07-04 02:19:04 +04:00
|
|
|
|
2001-05-02 01:33:43 +04:00
|
|
|
static struct DeadScript {
|
|
|
|
PRCList links;
|
|
|
|
JSDContext *jsdc;
|
|
|
|
jsdIScript *script;
|
2001-07-04 02:19:04 +04:00
|
|
|
} *gDeadScripts = nsnull;
|
|
|
|
|
2001-10-30 16:41:32 +03:00
|
|
|
enum PatternType {
|
|
|
|
ptIgnore = 0U,
|
|
|
|
ptStartsWith = 1U,
|
|
|
|
ptEndsWith = 2U,
|
|
|
|
ptContains = 3U,
|
|
|
|
ptEquals = 4U
|
|
|
|
};
|
|
|
|
|
|
|
|
static struct FilterRecord {
|
|
|
|
PRCList links;
|
|
|
|
jsdIFilter *filterObject;
|
|
|
|
void *glob;
|
|
|
|
char *urlPattern;
|
|
|
|
PRUint32 patternLength;
|
|
|
|
PatternType patternType;
|
|
|
|
PRUint32 startLine;
|
|
|
|
PRUint32 endLine;
|
|
|
|
} *gFilters = nsnull;
|
|
|
|
|
2001-07-04 02:19:04 +04:00
|
|
|
static struct LiveEphemeral *gLiveValues = nsnull, *gLiveProperties = nsnull;
|
|
|
|
|
|
|
|
/*******************************************************************************
|
|
|
|
* utility functions for ephemeral lists
|
|
|
|
*******************************************************************************/
|
|
|
|
|
|
|
|
void
|
|
|
|
jsds_InvalidateAllEphemerals (LiveEphemeral **listHead)
|
|
|
|
{
|
|
|
|
LiveEphemeral *lv_record =
|
|
|
|
NS_REINTERPRET_CAST (LiveEphemeral *,
|
|
|
|
PR_NEXT_LINK(&(*listHead)->links));
|
|
|
|
while (*listHead)
|
|
|
|
{
|
|
|
|
LiveEphemeral *next =
|
|
|
|
NS_REINTERPRET_CAST (LiveEphemeral *,
|
|
|
|
PR_NEXT_LINK(&lv_record->links));
|
|
|
|
lv_record->value->Invalidate();
|
|
|
|
lv_record = next;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
void
|
|
|
|
jsds_InsertEphemeral (LiveEphemeral **listHead, LiveEphemeral *item)
|
|
|
|
{
|
|
|
|
if (*listHead) {
|
|
|
|
/* if the list exists, add to it */
|
|
|
|
PR_APPEND_LINK(&item->links, &(*listHead)->links);
|
|
|
|
} else {
|
|
|
|
/* otherwise create the list */
|
|
|
|
PR_INIT_CLIST(&item->links);
|
|
|
|
*listHead = item;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
void
|
|
|
|
jsds_RemoveEphemeral (LiveEphemeral **listHead, LiveEphemeral *item)
|
|
|
|
{
|
|
|
|
LiveEphemeral *next =
|
|
|
|
NS_REINTERPRET_CAST (LiveEphemeral *,
|
|
|
|
PR_NEXT_LINK(&item->links));
|
|
|
|
|
|
|
|
if (next == item)
|
|
|
|
{
|
|
|
|
/* if the current item is also the next item, we're the only element,
|
|
|
|
* null out the list head */
|
|
|
|
NS_ASSERTION (*listHead == item,
|
|
|
|
"How could we not be the head of a one item list?");
|
|
|
|
*listHead = nsnull;
|
|
|
|
}
|
|
|
|
else if (item == *listHead)
|
|
|
|
{
|
|
|
|
/* otherwise, if we're currently the list head, change it */
|
|
|
|
*listHead = next;
|
|
|
|
}
|
|
|
|
|
|
|
|
PR_REMOVE_AND_INIT_LINK(&item->links);
|
|
|
|
}
|
2001-04-25 22:20:47 +04:00
|
|
|
|
2001-10-30 16:41:32 +03:00
|
|
|
/*******************************************************************************
|
|
|
|
* utility functions for filters
|
|
|
|
*******************************************************************************/
|
|
|
|
void
|
|
|
|
jsds_FreeFilter (FilterRecord *filter)
|
|
|
|
{
|
|
|
|
NS_IF_RELEASE (filter->filterObject);
|
|
|
|
if (filter->urlPattern)
|
|
|
|
nsMemory::Free(filter->urlPattern);
|
|
|
|
PR_Free (filter);
|
|
|
|
}
|
|
|
|
|
|
|
|
/* copies appropriate |filter| attributes into |rec|.
|
|
|
|
* False return indicates failure, the contents of |rec| will not be changed.
|
|
|
|
*/
|
|
|
|
PRBool
|
|
|
|
jsds_SyncFilter (FilterRecord *rec, jsdIFilter *filter)
|
|
|
|
{
|
|
|
|
NS_ASSERTION (rec, "jsds_SyncFilter without rec");
|
|
|
|
NS_ASSERTION (filter, "jsds_SyncFilter without filter");
|
|
|
|
|
|
|
|
JSObject *glob_proper = nsnull;
|
|
|
|
nsCOMPtr<nsISupports> glob;
|
|
|
|
nsresult rv = filter->GetGlob(getter_AddRefs(glob));
|
|
|
|
if (NS_FAILED(rv))
|
|
|
|
return PR_FALSE;
|
|
|
|
if (glob) {
|
|
|
|
nsCOMPtr<nsIScriptGlobalObject> nsiglob = do_QueryInterface(glob);
|
|
|
|
if (nsiglob)
|
|
|
|
glob_proper = nsiglob->GetGlobalJSObject();
|
|
|
|
}
|
|
|
|
|
|
|
|
PRUint32 startLine;
|
|
|
|
rv = filter->GetStartLine(&startLine);
|
|
|
|
if (NS_FAILED(rv))
|
|
|
|
return PR_FALSE;
|
|
|
|
|
|
|
|
PRUint32 endLine;
|
|
|
|
rv = filter->GetStartLine(&endLine);
|
|
|
|
if (NS_FAILED(rv))
|
|
|
|
return PR_FALSE;
|
|
|
|
|
|
|
|
char *urlPattern;
|
|
|
|
rv = filter->GetUrlPattern (&urlPattern);
|
|
|
|
if (NS_FAILED(rv))
|
|
|
|
return PR_FALSE;
|
|
|
|
|
|
|
|
if (urlPattern) {
|
|
|
|
PRUint32 len = PL_strlen(urlPattern);
|
|
|
|
if (urlPattern[0] == '*') {
|
|
|
|
/* pattern starts with a *, shift all chars once to the left,
|
|
|
|
* including the trailing null. */
|
|
|
|
memmove (&urlPattern[0], &urlPattern[1], len);
|
|
|
|
|
|
|
|
if (urlPattern[len - 2] == '*') {
|
|
|
|
/* pattern is in the format "*foo*", overwrite the final * with
|
|
|
|
* a null. */
|
|
|
|
urlPattern[len - 2] = '\0';
|
|
|
|
rec->patternType = ptContains;
|
|
|
|
rec->patternLength = len - 2;
|
|
|
|
} else {
|
|
|
|
/* pattern is in the format "*foo", just make a note of the
|
|
|
|
* new length. */
|
|
|
|
rec->patternType = ptEndsWith;
|
|
|
|
rec->patternLength = len - 1;
|
|
|
|
}
|
|
|
|
} else if (urlPattern[len - 1] == '*') {
|
|
|
|
/* pattern is in the format "foo*", overwrite the final * with a
|
|
|
|
* null. */
|
|
|
|
urlPattern[len - 1] = '\0';
|
|
|
|
rec->patternType = ptStartsWith;
|
|
|
|
rec->patternLength = len - 1;
|
|
|
|
} else {
|
|
|
|
/* pattern is in the format "foo". */
|
|
|
|
rec->patternType = ptEquals;
|
|
|
|
rec->patternLength = len;
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
rec->patternType = ptIgnore;
|
|
|
|
rec->patternLength = 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* we got everything we need without failing, now copy it into rec. */
|
|
|
|
|
|
|
|
if (rec->filterObject != filter) {
|
|
|
|
NS_IF_RELEASE(rec->filterObject);
|
|
|
|
NS_ADDREF(filter);
|
|
|
|
rec->filterObject = filter;
|
|
|
|
}
|
|
|
|
|
|
|
|
rec->glob = glob_proper;
|
|
|
|
|
|
|
|
rec->startLine = startLine;
|
|
|
|
rec->endLine = endLine;
|
|
|
|
|
|
|
|
if (rec->urlPattern)
|
|
|
|
nsMemory::Free (rec->urlPattern);
|
|
|
|
rec->urlPattern = urlPattern;
|
|
|
|
|
|
|
|
return PR_TRUE;
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
FilterRecord *
|
|
|
|
jsds_FindFilter (jsdIFilter *filter)
|
|
|
|
{
|
|
|
|
if (!gFilters)
|
|
|
|
return nsnull;
|
|
|
|
|
|
|
|
FilterRecord *current = gFilters;
|
|
|
|
|
|
|
|
do {
|
|
|
|
if (current->filterObject == filter)
|
|
|
|
return current;
|
|
|
|
current = NS_REINTERPRET_CAST(FilterRecord *,
|
|
|
|
PR_NEXT_LINK(¤t->links));
|
|
|
|
} while (current != gFilters);
|
|
|
|
|
|
|
|
return nsnull;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* returns true if the hook should be executed. */
|
|
|
|
PRBool
|
|
|
|
jsds_FilterHook (JSDContext *jsdc, JSDThreadState *state)
|
|
|
|
{
|
|
|
|
if (!gFilters)
|
|
|
|
return PR_TRUE;
|
|
|
|
|
|
|
|
JSContext *cx = JSD_GetJSContext (jsdc, state);
|
|
|
|
void *glob = NS_STATIC_CAST(void *, JS_GetGlobalObject (cx));
|
|
|
|
if (NS_WARN_IF_FALSE(glob, "No global in threadstate"))
|
|
|
|
return PR_TRUE;
|
|
|
|
|
|
|
|
JSDStackFrameInfo *frame = JSD_GetStackFrame (jsdc, state);
|
|
|
|
if (NS_WARN_IF_FALSE(frame, "No frame in threadstate"))
|
|
|
|
return PR_TRUE;
|
|
|
|
|
|
|
|
JSDScript *script = JSD_GetScriptForStackFrame (jsdc, state, frame);
|
|
|
|
if (NS_WARN_IF_FALSE(script, "No script in threadstate"))
|
|
|
|
return PR_TRUE;
|
|
|
|
|
|
|
|
jsuint pc = JSD_GetPCForStackFrame (jsdc, state, frame);
|
|
|
|
if (NS_WARN_IF_FALSE(pc, "No pc in threadstate"))
|
|
|
|
return PR_TRUE;
|
|
|
|
|
|
|
|
PRUint32 currentLine = JSD_GetClosestLine (jsdc, script, pc);
|
|
|
|
if (NS_WARN_IF_FALSE(currentLine, "Can't convert pc to line"))
|
|
|
|
return PR_TRUE;
|
|
|
|
|
|
|
|
const char *url = nsnull;
|
|
|
|
PRUint32 len = 0;
|
|
|
|
FilterRecord *currentFilter = gFilters;
|
|
|
|
do {
|
|
|
|
PRUint32 flags = 0;
|
|
|
|
nsresult rv = currentFilter->filterObject->GetFlags(&flags);
|
|
|
|
NS_ASSERTION(NS_SUCCEEDED(rv), "Error getting flags for filter");
|
|
|
|
if (flags & jsdIFilter::FLAG_ENABLED) {
|
|
|
|
/* if there is no glob, or the globs match */
|
|
|
|
if ((!currentFilter->glob || currentFilter->glob == glob) &&
|
|
|
|
/* and there is no start line, or the start line is before
|
|
|
|
* or equal to the current */
|
|
|
|
(!currentFilter->startLine ||
|
|
|
|
currentFilter->startLine <= currentLine) &&
|
|
|
|
/* and there is no end line, or the end line is after
|
|
|
|
* or equal to the current */
|
|
|
|
(!currentFilter->endLine ||
|
|
|
|
currentFilter->endLine >= currentLine)) {
|
|
|
|
/* then we're going to have to compare the url. */
|
|
|
|
if (currentFilter->patternType == ptIgnore)
|
|
|
|
return flags & jsdIFilter::FLAG_PASS;
|
|
|
|
|
|
|
|
if (!url) {
|
|
|
|
url = JSD_GetScriptFilename (jsdc, script);
|
|
|
|
NS_ASSERTION (url, "Script with no filename");
|
|
|
|
len = PL_strlen(url);
|
|
|
|
}
|
|
|
|
|
|
|
|
if (len >= currentFilter->patternLength) {
|
|
|
|
switch (currentFilter->patternType) {
|
|
|
|
case ptEquals:
|
|
|
|
if (!PL_strcmp(currentFilter->urlPattern, url))
|
|
|
|
return flags & jsdIFilter::FLAG_PASS;
|
|
|
|
break;
|
|
|
|
case ptStartsWith:
|
|
|
|
if (!PL_strncmp(currentFilter->urlPattern, url,
|
|
|
|
currentFilter->patternLength))
|
|
|
|
return flags & jsdIFilter::FLAG_PASS;
|
|
|
|
break;
|
|
|
|
case ptEndsWith:
|
|
|
|
if (!PL_strcmp(currentFilter->urlPattern,
|
|
|
|
&url[len -
|
|
|
|
currentFilter->patternLength]))
|
|
|
|
return flags & jsdIFilter::FLAG_PASS;
|
|
|
|
break;
|
|
|
|
case ptContains:
|
|
|
|
if (PL_strstr(url, currentFilter->urlPattern))
|
|
|
|
return flags & jsdIFilter::FLAG_PASS;
|
|
|
|
break;
|
|
|
|
default:
|
|
|
|
NS_ASSERTION(0, "Invalid pattern type");
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
currentFilter = NS_REINTERPRET_CAST(FilterRecord *,
|
|
|
|
PR_NEXT_LINK(¤tFilter->links));
|
|
|
|
} while (currentFilter != gFilters);
|
|
|
|
|
|
|
|
return PR_TRUE;
|
|
|
|
|
|
|
|
}
|
|
|
|
|
2001-04-25 22:20:47 +04:00
|
|
|
/*******************************************************************************
|
|
|
|
* c callbacks
|
|
|
|
*******************************************************************************/
|
|
|
|
|
2001-09-22 05:39:24 +04:00
|
|
|
JS_STATIC_DLL_CALLBACK (void)
|
2001-08-29 02:02:07 +04:00
|
|
|
jsds_NotifyPendingDeadScripts (JSContext *cx)
|
2001-06-28 11:46:36 +04:00
|
|
|
{
|
|
|
|
nsCOMPtr<jsdIScriptHook> hook = 0;
|
|
|
|
gJsds->GetScriptHook (getter_AddRefs(hook));
|
|
|
|
if (hook)
|
|
|
|
{
|
|
|
|
DeadScript *ds;
|
2001-08-29 02:02:07 +04:00
|
|
|
#ifdef CAUTIOUS_SCRIPTHOOK
|
|
|
|
JSRuntime *rt = JS_GetRuntime(cx);
|
|
|
|
#endif
|
2001-10-30 16:41:32 +03:00
|
|
|
gJsds->Pause(nsnull);
|
2001-06-28 11:46:36 +04:00
|
|
|
do {
|
|
|
|
ds = gDeadScripts;
|
|
|
|
|
|
|
|
/* tell the user this script has been destroyed */
|
2001-08-29 02:02:07 +04:00
|
|
|
#ifdef CAUTIOUS_SCRIPTHOOK
|
|
|
|
JS_DISABLE_GC(rt);
|
|
|
|
#endif
|
2001-06-28 11:46:36 +04:00
|
|
|
hook->OnScriptDestroyed (ds->script);
|
2001-08-29 02:02:07 +04:00
|
|
|
#ifdef CAUTIOUS_SCRIPTHOOK
|
|
|
|
JS_ENABLE_GC(rt);
|
|
|
|
#endif
|
2001-06-28 11:46:36 +04:00
|
|
|
/* get next deleted script */
|
|
|
|
gDeadScripts = NS_REINTERPRET_CAST(DeadScript *,
|
|
|
|
PR_NEXT_LINK(&ds->links));
|
|
|
|
/* take ourselves out of the circular list */
|
|
|
|
PR_REMOVE_LINK(&ds->links);
|
|
|
|
/* addref came from the FromPtr call in jsds_ScriptHookProc */
|
|
|
|
NS_RELEASE(ds->script);
|
|
|
|
/* free the struct! */
|
|
|
|
PR_Free(ds);
|
|
|
|
} while (&gDeadScripts->links != &ds->links);
|
|
|
|
/* keep going until we catch up with our tail */
|
2001-10-30 16:41:32 +03:00
|
|
|
gJsds->UnPause(nsnull);
|
2001-06-28 11:46:36 +04:00
|
|
|
}
|
|
|
|
gDeadScripts = 0;
|
|
|
|
}
|
|
|
|
|
2001-09-22 05:39:24 +04:00
|
|
|
JS_STATIC_DLL_CALLBACK (JSBool)
|
2001-04-26 01:54:59 +04:00
|
|
|
jsds_GCCallbackProc (JSContext *cx, JSGCStatus status)
|
2001-04-25 22:20:47 +04:00
|
|
|
{
|
2001-04-26 01:54:59 +04:00
|
|
|
gGCStatus = status;
|
2001-10-30 16:41:32 +03:00
|
|
|
#ifdef DEBUG_verbose
|
2001-04-26 01:54:59 +04:00
|
|
|
printf ("new gc status is %i\n", status);
|
2001-05-02 01:33:43 +04:00
|
|
|
#endif
|
2001-06-28 11:46:36 +04:00
|
|
|
if (status == JSGC_END && gDeadScripts)
|
2001-08-29 02:02:07 +04:00
|
|
|
jsds_NotifyPendingDeadScripts (cx);
|
2001-05-02 01:33:43 +04:00
|
|
|
|
2001-04-26 01:54:59 +04:00
|
|
|
if (gLastGCProc)
|
|
|
|
return gLastGCProc (cx, status);
|
|
|
|
|
|
|
|
return JS_TRUE;
|
2001-04-25 22:20:47 +04:00
|
|
|
}
|
|
|
|
|
2001-09-22 05:39:24 +04:00
|
|
|
JS_STATIC_DLL_CALLBACK (JSBool)
|
2001-09-01 22:03:53 +04:00
|
|
|
jsds_CallHookProc (JSDContext* jsdc, JSDThreadState* jsdthreadstate,
|
|
|
|
uintN type, void* callerdata)
|
|
|
|
{
|
|
|
|
nsCOMPtr<jsdICallHook> hook(0);
|
|
|
|
|
|
|
|
switch (type)
|
|
|
|
{
|
|
|
|
case JSD_HOOK_TOPLEVEL_START:
|
|
|
|
case JSD_HOOK_TOPLEVEL_END:
|
|
|
|
gJsds->GetTopLevelHook(getter_AddRefs(hook));
|
|
|
|
break;
|
|
|
|
|
|
|
|
case JSD_HOOK_FUNCTION_CALL:
|
|
|
|
case JSD_HOOK_FUNCTION_RETURN:
|
|
|
|
gJsds->GetFunctionHook(getter_AddRefs(hook));
|
|
|
|
break;
|
|
|
|
|
|
|
|
default:
|
|
|
|
NS_ASSERTION (0, "Unknown hook type.");
|
|
|
|
}
|
|
|
|
|
|
|
|
if (!hook)
|
|
|
|
return JS_TRUE;
|
|
|
|
|
2001-10-30 16:41:32 +03:00
|
|
|
if (!jsds_FilterHook (jsdc, jsdthreadstate))
|
|
|
|
return JS_FALSE;
|
|
|
|
|
2001-09-01 22:03:53 +04:00
|
|
|
JSDStackFrameInfo *native_frame = JSD_GetStackFrame (jsdc, jsdthreadstate);
|
|
|
|
nsCOMPtr<jsdIStackFrame> frame =
|
|
|
|
getter_AddRefs(jsdStackFrame::FromPtr(jsdc, jsdthreadstate,
|
|
|
|
native_frame));
|
2001-10-30 16:41:32 +03:00
|
|
|
gJsds->Pause(nsnull);
|
2001-09-01 22:03:53 +04:00
|
|
|
hook->OnCall(frame, type);
|
2001-10-30 16:41:32 +03:00
|
|
|
gJsds->UnPause(nsnull);
|
2001-09-01 22:03:53 +04:00
|
|
|
frame->Invalidate();
|
|
|
|
|
|
|
|
return JS_TRUE;
|
|
|
|
}
|
|
|
|
|
2001-09-22 05:39:24 +04:00
|
|
|
JS_STATIC_DLL_CALLBACK (PRUint32)
|
2001-05-05 02:14:04 +04:00
|
|
|
jsds_ExecutionHookProc (JSDContext* jsdc, JSDThreadState* jsdthreadstate,
|
|
|
|
uintN type, void* callerdata, jsval* rval)
|
2001-04-25 22:20:47 +04:00
|
|
|
{
|
2001-05-05 02:14:04 +04:00
|
|
|
nsCOMPtr<jsdIExecutionHook> hook(0);
|
2001-04-25 22:20:47 +04:00
|
|
|
PRUint32 hook_rv = JSD_HOOK_RETURN_CONTINUE;
|
2001-05-05 02:14:04 +04:00
|
|
|
jsdIValue *js_rv = 0;
|
2001-04-25 22:20:47 +04:00
|
|
|
|
2001-05-05 02:14:04 +04:00
|
|
|
switch (type)
|
2001-04-25 22:20:47 +04:00
|
|
|
{
|
2001-05-05 02:14:04 +04:00
|
|
|
case JSD_HOOK_INTERRUPTED:
|
|
|
|
gJsds->GetInterruptHook(getter_AddRefs(hook));
|
|
|
|
break;
|
|
|
|
case JSD_HOOK_DEBUG_REQUESTED:
|
|
|
|
gJsds->GetErrorHook(getter_AddRefs(hook));
|
|
|
|
break;
|
|
|
|
case JSD_HOOK_DEBUGGER_KEYWORD:
|
|
|
|
gJsds->GetDebuggerHook(getter_AddRefs(hook));
|
|
|
|
break;
|
|
|
|
case JSD_HOOK_BREAKPOINT:
|
|
|
|
gJsds->GetBreakpointHook(getter_AddRefs(hook));
|
|
|
|
break;
|
|
|
|
case JSD_HOOK_THROW:
|
|
|
|
{
|
2001-09-01 22:03:53 +04:00
|
|
|
hook_rv = JSD_HOOK_RETURN_CONTINUE_THROW;
|
2001-05-05 02:14:04 +04:00
|
|
|
gJsds->GetThrowHook(getter_AddRefs(hook));
|
|
|
|
if (hook) {
|
|
|
|
JSDValue *jsdv = JSD_GetException (jsdc, jsdthreadstate);
|
|
|
|
js_rv = jsdValue::FromPtr (jsdc, jsdv);
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
default:
|
|
|
|
NS_ASSERTION (0, "Unknown hook type.");
|
2001-04-25 22:20:47 +04:00
|
|
|
}
|
|
|
|
|
2001-05-05 02:14:04 +04:00
|
|
|
if (!hook)
|
2001-09-01 22:03:53 +04:00
|
|
|
return hook_rv;
|
2001-04-25 22:20:47 +04:00
|
|
|
|
2001-10-30 16:41:32 +03:00
|
|
|
if (!jsds_FilterHook (jsdc, jsdthreadstate))
|
|
|
|
return JSD_HOOK_RETURN_CONTINUE;
|
|
|
|
|
2001-07-04 02:19:04 +04:00
|
|
|
JSDStackFrameInfo *native_frame = JSD_GetStackFrame (jsdc, jsdthreadstate);
|
|
|
|
nsCOMPtr<jsdIStackFrame> frame =
|
|
|
|
getter_AddRefs(jsdStackFrame::FromPtr(jsdc, jsdthreadstate,
|
|
|
|
native_frame));
|
2001-10-30 16:41:32 +03:00
|
|
|
gJsds->Pause(nsnull);
|
2001-07-04 02:19:04 +04:00
|
|
|
hook->OnExecute (frame, type, &js_rv, &hook_rv);
|
2001-10-30 16:41:32 +03:00
|
|
|
gJsds->UnPause(nsnull);
|
2001-07-04 02:19:04 +04:00
|
|
|
frame->Invalidate();
|
|
|
|
|
2001-05-05 02:14:04 +04:00
|
|
|
if (hook_rv == JSD_HOOK_RETURN_RET_WITH_VAL ||
|
|
|
|
hook_rv == JSD_HOOK_RETURN_THROW_WITH_VAL)
|
2001-04-25 22:20:47 +04:00
|
|
|
{
|
|
|
|
JSDValue *jsdv;
|
|
|
|
js_rv->GetJSDValue (&jsdv);
|
|
|
|
*rval = JSD_GetValueWrappedJSVal(jsdc, jsdv);
|
|
|
|
}
|
2001-05-05 02:14:04 +04:00
|
|
|
|
|
|
|
NS_IF_RELEASE(js_rv);
|
2001-04-25 22:20:47 +04:00
|
|
|
return hook_rv;
|
|
|
|
}
|
|
|
|
|
2001-09-22 05:39:24 +04:00
|
|
|
JS_STATIC_DLL_CALLBACK (void)
|
2001-04-25 22:20:47 +04:00
|
|
|
jsds_ScriptHookProc (JSDContext* jsdc, JSDScript* jsdscript, JSBool creating,
|
|
|
|
void* callerdata)
|
|
|
|
{
|
2001-08-29 02:02:07 +04:00
|
|
|
#ifdef CAUTIOUS_SCRIPTHOOK
|
|
|
|
JSContext *cx = JSD_GetDefaultJSContext(jsdc);
|
|
|
|
JSRuntime *rt = JS_GetRuntime(cx);
|
|
|
|
#endif
|
|
|
|
|
2001-05-02 01:33:43 +04:00
|
|
|
if (creating) {
|
|
|
|
jsdIScriptHook *hook = 0;
|
|
|
|
|
|
|
|
gJsds->GetScriptHook (&hook);
|
|
|
|
if (!hook)
|
|
|
|
return;
|
|
|
|
|
|
|
|
nsCOMPtr<jsdIScript> script =
|
|
|
|
getter_AddRefs(jsdScript::FromPtr(jsdc, jsdscript));
|
2001-08-29 02:02:07 +04:00
|
|
|
#ifdef CAUTIOUS_SCRIPTHOOK
|
|
|
|
JS_DISABLE_GC(rt);
|
|
|
|
#endif
|
2001-10-30 16:41:32 +03:00
|
|
|
gJsds->Pause(nsnull);
|
2001-05-05 02:14:04 +04:00
|
|
|
hook->OnScriptCreated (script);
|
2001-10-30 16:41:32 +03:00
|
|
|
gJsds->UnPause(nsnull);
|
2001-08-29 02:02:07 +04:00
|
|
|
#ifdef CAUTIOUS_SCRIPTHOOK
|
|
|
|
JS_ENABLE_GC(rt);
|
|
|
|
#endif
|
2001-05-02 01:33:43 +04:00
|
|
|
} else {
|
2001-06-28 11:46:36 +04:00
|
|
|
|
2001-05-02 01:33:43 +04:00
|
|
|
jsdIScript *jsdis = jsdScript::FromPtr(jsdc, jsdscript);
|
|
|
|
/* the initial addref is owned by the DeadScript record */
|
|
|
|
jsdis->Invalidate();
|
|
|
|
|
|
|
|
if (gGCStatus == JSGC_END) {
|
|
|
|
/* if GC *isn't* running, we can tell the user about the script
|
|
|
|
* delete now. */
|
|
|
|
nsCOMPtr<jsdIScriptHook> hook = 0;
|
|
|
|
gJsds->GetScriptHook (getter_AddRefs(hook));
|
|
|
|
if (hook) {
|
2001-08-29 02:02:07 +04:00
|
|
|
#ifdef CAUTIOUS_SCRIPTHOOK
|
|
|
|
JS_DISABLE_GC(rt);
|
|
|
|
#endif
|
2001-10-30 16:41:32 +03:00
|
|
|
|
|
|
|
gJsds->Pause(nsnull);
|
2001-05-05 02:14:04 +04:00
|
|
|
hook->OnScriptDestroyed (jsdis);
|
2001-10-30 16:41:32 +03:00
|
|
|
gJsds->UnPause(nsnull);
|
2001-08-29 02:02:07 +04:00
|
|
|
#ifdef CAUTIOUS_SCRIPTHOOK
|
|
|
|
JS_ENABLE_GC(rt);
|
|
|
|
#endif
|
2001-05-02 01:33:43 +04:00
|
|
|
}
|
|
|
|
} else {
|
|
|
|
/* if a GC *is* running, we've got to wait until it's done before
|
|
|
|
* we can execute any JS, so we queue the notification in a PRCList
|
|
|
|
* until GC tells us it's done. See jsds_GCCallbackProc(). */
|
|
|
|
DeadScript *ds = PR_NEW(DeadScript);
|
|
|
|
if (!ds) {
|
|
|
|
NS_RELEASE(jsdis);
|
|
|
|
return; /* NS_ERROR_OUT_OF_MEMORY */
|
|
|
|
}
|
|
|
|
|
|
|
|
ds->jsdc = jsdc;
|
|
|
|
ds->script = jsdis;
|
|
|
|
|
|
|
|
if (gDeadScripts)
|
|
|
|
/* if the queue exists, add to it */
|
|
|
|
PR_APPEND_LINK(&ds->links, &gDeadScripts->links);
|
|
|
|
else {
|
|
|
|
/* otherwise create the queue */
|
|
|
|
PR_INIT_CLIST(&ds->links);
|
|
|
|
gDeadScripts = ds;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
2001-04-25 22:20:47 +04:00
|
|
|
|
2001-05-02 01:33:43 +04:00
|
|
|
|
2001-04-25 22:20:47 +04:00
|
|
|
}
|
|
|
|
|
2001-04-17 12:50:41 +04:00
|
|
|
/*******************************************************************************
|
|
|
|
* reflected jsd data structures
|
|
|
|
*******************************************************************************/
|
|
|
|
|
2001-04-20 07:54:05 +04:00
|
|
|
/* Contexts */
|
2001-05-05 02:14:04 +04:00
|
|
|
/*
|
2001-04-17 12:50:41 +04:00
|
|
|
NS_IMPL_THREADSAFE_ISUPPORTS1(jsdContext, jsdIContext);
|
|
|
|
|
|
|
|
NS_IMETHODIMP
|
|
|
|
jsdContext::GetJSDContext(JSDContext **_rval)
|
|
|
|
{
|
|
|
|
*_rval = mCx;
|
|
|
|
return NS_OK;
|
|
|
|
}
|
2001-05-05 02:14:04 +04:00
|
|
|
*/
|
2001-04-17 12:50:41 +04:00
|
|
|
|
2001-04-20 07:54:05 +04:00
|
|
|
/* Objects */
|
|
|
|
NS_IMPL_THREADSAFE_ISUPPORTS1(jsdObject, jsdIObject);
|
|
|
|
|
|
|
|
NS_IMETHODIMP
|
|
|
|
jsdObject::GetJSDContext(JSDContext **_rval)
|
|
|
|
{
|
|
|
|
*_rval = mCx;
|
|
|
|
return NS_OK;
|
|
|
|
}
|
|
|
|
|
|
|
|
NS_IMETHODIMP
|
|
|
|
jsdObject::GetJSDObject(JSDObject **_rval)
|
|
|
|
{
|
|
|
|
*_rval = mObject;
|
|
|
|
return NS_OK;
|
|
|
|
}
|
|
|
|
|
2001-04-25 22:20:47 +04:00
|
|
|
NS_IMETHODIMP
|
|
|
|
jsdObject::GetCreatorURL(char **_rval)
|
|
|
|
{
|
2001-09-29 12:28:41 +04:00
|
|
|
*_rval = ToNewCString(nsDependentCString(JSD_GetObjectNewURL(mCx, mObject)));
|
2001-04-25 22:20:47 +04:00
|
|
|
return NS_OK;
|
|
|
|
}
|
|
|
|
|
|
|
|
NS_IMETHODIMP
|
|
|
|
jsdObject::GetCreatorLine(PRUint32 *_rval)
|
|
|
|
{
|
|
|
|
*_rval = JSD_GetObjectNewLineNumber(mCx, mObject);
|
|
|
|
return NS_OK;
|
|
|
|
}
|
|
|
|
|
|
|
|
NS_IMETHODIMP
|
|
|
|
jsdObject::GetConstructorURL(char **_rval)
|
|
|
|
{
|
2001-09-29 12:28:41 +04:00
|
|
|
*_rval = ToNewCString(nsDependentCString(JSD_GetObjectConstructorURL(mCx, mObject)));
|
2001-04-25 22:20:47 +04:00
|
|
|
return NS_OK;
|
|
|
|
}
|
|
|
|
|
|
|
|
NS_IMETHODIMP
|
|
|
|
jsdObject::GetConstructorLine(PRUint32 *_rval)
|
|
|
|
{
|
|
|
|
*_rval = JSD_GetObjectConstructorLineNumber(mCx, mObject);
|
|
|
|
return NS_OK;
|
|
|
|
}
|
|
|
|
|
|
|
|
NS_IMETHODIMP
|
|
|
|
jsdObject::GetValue(jsdIValue **_rval)
|
|
|
|
{
|
|
|
|
JSDValue *jsdv = JSD_GetValueForObject (mCx, mObject);
|
|
|
|
|
|
|
|
*_rval = jsdValue::FromPtr (mCx, jsdv);
|
|
|
|
return NS_OK;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* PC */
|
|
|
|
NS_IMPL_THREADSAFE_ISUPPORTS1(jsdPC, jsdIPC);
|
|
|
|
|
|
|
|
NS_IMETHODIMP
|
|
|
|
jsdPC::GetPc(jsuword *_rval)
|
|
|
|
{
|
|
|
|
*_rval = mPC;
|
|
|
|
return NS_OK;
|
|
|
|
}
|
|
|
|
|
2001-04-20 07:54:05 +04:00
|
|
|
/* Properties */
|
2001-07-04 02:19:04 +04:00
|
|
|
NS_IMPL_THREADSAFE_ISUPPORTS2(jsdProperty, jsdIProperty, jsdIEphemeral);
|
|
|
|
|
|
|
|
jsdProperty::jsdProperty (JSDContext *aCx, JSDProperty *aProperty) :
|
|
|
|
mCx(aCx), mProperty(aProperty)
|
|
|
|
{
|
|
|
|
DEBUG_CREATE ("jsdProperty", gPropertyCount);
|
|
|
|
mValid = (aCx && aProperty);
|
|
|
|
NS_INIT_ISUPPORTS();
|
|
|
|
mLiveListEntry.value = this;
|
|
|
|
jsds_InsertEphemeral (&gLiveProperties, &mLiveListEntry);
|
|
|
|
}
|
|
|
|
|
|
|
|
jsdProperty::~jsdProperty ()
|
|
|
|
{
|
|
|
|
DEBUG_DESTROY ("jsdProperty", gPropertyCount);
|
|
|
|
if (mValid)
|
|
|
|
Invalidate();
|
|
|
|
}
|
|
|
|
|
|
|
|
NS_IMETHODIMP
|
|
|
|
jsdProperty::Invalidate()
|
|
|
|
{
|
|
|
|
ASSERT_VALID_VALUE;
|
|
|
|
mValid = PR_FALSE;
|
|
|
|
jsds_RemoveEphemeral (&gLiveProperties, &mLiveListEntry);
|
|
|
|
JSD_DropProperty (mCx, mProperty);
|
|
|
|
return NS_OK;
|
|
|
|
}
|
|
|
|
|
|
|
|
void
|
|
|
|
jsdProperty::InvalidateAll()
|
|
|
|
{
|
|
|
|
if (gLiveProperties)
|
|
|
|
jsds_InvalidateAllEphemerals (&gLiveProperties);
|
|
|
|
}
|
2001-04-20 07:54:05 +04:00
|
|
|
|
|
|
|
NS_IMETHODIMP
|
|
|
|
jsdProperty::GetJSDContext(JSDContext **_rval)
|
|
|
|
{
|
|
|
|
*_rval = mCx;
|
|
|
|
return NS_OK;
|
|
|
|
}
|
|
|
|
|
|
|
|
NS_IMETHODIMP
|
|
|
|
jsdProperty::GetJSDProperty(JSDProperty **_rval)
|
|
|
|
{
|
|
|
|
*_rval = mProperty;
|
|
|
|
return NS_OK;
|
|
|
|
}
|
|
|
|
|
2001-07-04 02:19:04 +04:00
|
|
|
NS_IMETHODIMP
|
|
|
|
jsdProperty::GetIsValid(PRBool *_rval)
|
|
|
|
{
|
|
|
|
*_rval = mValid;
|
|
|
|
return NS_OK;
|
|
|
|
}
|
|
|
|
|
2001-04-21 11:21:27 +04:00
|
|
|
NS_IMETHODIMP
|
|
|
|
jsdProperty::GetAlias(jsdIValue **_rval)
|
|
|
|
{
|
|
|
|
JSDValue *jsdv = JSD_GetPropertyValue (mCx, mProperty);
|
|
|
|
|
|
|
|
*_rval = jsdValue::FromPtr (mCx, jsdv);
|
|
|
|
return NS_OK;
|
|
|
|
}
|
|
|
|
|
|
|
|
NS_IMETHODIMP
|
|
|
|
jsdProperty::GetFlags(PRUint32 *_rval)
|
|
|
|
{
|
|
|
|
*_rval = JSD_GetPropertyFlags (mCx, mProperty);
|
|
|
|
return NS_OK;
|
|
|
|
}
|
|
|
|
|
|
|
|
NS_IMETHODIMP
|
|
|
|
jsdProperty::GetName(jsdIValue **_rval)
|
|
|
|
{
|
|
|
|
JSDValue *jsdv = JSD_GetPropertyName (mCx, mProperty);
|
|
|
|
|
|
|
|
*_rval = jsdValue::FromPtr (mCx, jsdv);
|
|
|
|
return NS_OK;
|
|
|
|
}
|
|
|
|
|
|
|
|
NS_IMETHODIMP
|
|
|
|
jsdProperty::GetValue(jsdIValue **_rval)
|
|
|
|
{
|
|
|
|
JSDValue *jsdv = JSD_GetPropertyValue (mCx, mProperty);
|
|
|
|
|
|
|
|
*_rval = jsdValue::FromPtr (mCx, jsdv);
|
|
|
|
return NS_OK;
|
|
|
|
}
|
|
|
|
|
|
|
|
NS_IMETHODIMP
|
|
|
|
jsdProperty::GetVarArgSlot(PRUint32 *_rval)
|
|
|
|
{
|
|
|
|
*_rval = JSD_GetPropertyVarArgSlot (mCx, mProperty);
|
|
|
|
return NS_OK;
|
|
|
|
}
|
|
|
|
|
2001-04-20 07:54:05 +04:00
|
|
|
/* Scripts */
|
2001-07-04 02:19:04 +04:00
|
|
|
NS_IMPL_THREADSAFE_ISUPPORTS2(jsdScript, jsdIScript, jsdIEphemeral);
|
|
|
|
|
|
|
|
jsdScript::jsdScript (JSDContext *aCx, JSDScript *aScript) : mValid(PR_FALSE),
|
|
|
|
mCx(aCx),
|
|
|
|
mScript(aScript),
|
|
|
|
mFileName(0),
|
|
|
|
mFunctionName(0),
|
|
|
|
mBaseLineNumber(0),
|
|
|
|
mLineExtent(0)
|
|
|
|
{
|
|
|
|
DEBUG_CREATE ("jsdScript", gScriptCount);
|
|
|
|
NS_INIT_ISUPPORTS();
|
|
|
|
|
2001-10-30 16:41:32 +03:00
|
|
|
if (mScript) {
|
2001-07-04 02:19:04 +04:00
|
|
|
/* copy the script's information now, so we have it later, when it
|
|
|
|
* gets destroyed. */
|
|
|
|
JSD_LockScriptSubsystem(mCx);
|
|
|
|
mFileName = new nsCString(JSD_GetScriptFilename(mCx, mScript));
|
|
|
|
mFunctionName =
|
|
|
|
new nsCString(JSD_GetScriptFunctionName(mCx, mScript));
|
|
|
|
mBaseLineNumber = JSD_GetScriptBaseLineNumber(mCx, mScript);
|
|
|
|
mLineExtent = JSD_GetScriptLineExtent(mCx, mScript);
|
|
|
|
JSD_UnlockScriptSubsystem(mCx);
|
|
|
|
|
2001-09-22 05:39:24 +04:00
|
|
|
mValid = PR_TRUE;
|
2001-07-04 02:19:04 +04:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
jsdScript::~jsdScript ()
|
|
|
|
{
|
|
|
|
DEBUG_DESTROY ("jsdScript", gScriptCount);
|
|
|
|
if (mFileName)
|
|
|
|
delete mFileName;
|
|
|
|
if (mFunctionName)
|
|
|
|
delete mFunctionName;
|
|
|
|
|
|
|
|
/* Invalidate() needs to be called to release an owning reference to
|
|
|
|
* ourselves, so if we got here without being invalidated, something
|
|
|
|
* has gone wrong with our ref count. */
|
|
|
|
NS_ASSERTION (!mValid, "Script destroyed without being invalidated.");
|
|
|
|
|
|
|
|
}
|
2001-04-17 12:50:41 +04:00
|
|
|
|
2001-04-20 07:54:05 +04:00
|
|
|
NS_IMETHODIMP
|
|
|
|
jsdScript::GetJSDContext(JSDContext **_rval)
|
|
|
|
{
|
2001-05-02 01:33:43 +04:00
|
|
|
ASSERT_VALID_SCRIPT;
|
2001-04-20 07:54:05 +04:00
|
|
|
*_rval = mCx;
|
|
|
|
return NS_OK;
|
|
|
|
}
|
|
|
|
|
2001-04-17 12:50:41 +04:00
|
|
|
NS_IMETHODIMP
|
|
|
|
jsdScript::GetJSDScript(JSDScript **_rval)
|
|
|
|
{
|
2001-05-02 01:33:43 +04:00
|
|
|
ASSERT_VALID_SCRIPT;
|
2001-04-17 12:50:41 +04:00
|
|
|
*_rval = mScript;
|
|
|
|
return NS_OK;
|
|
|
|
}
|
|
|
|
|
2001-05-02 01:33:43 +04:00
|
|
|
NS_IMETHODIMP
|
|
|
|
jsdScript::Invalidate()
|
|
|
|
{
|
|
|
|
ASSERT_VALID_SCRIPT;
|
2001-06-28 11:46:36 +04:00
|
|
|
mValid = PR_FALSE;
|
2001-05-02 01:33:43 +04:00
|
|
|
|
|
|
|
/* release the addref we do in FromPtr */
|
|
|
|
jsdIScript *script = NS_STATIC_CAST(jsdIScript *,
|
|
|
|
JSD_GetScriptPrivate(mScript));
|
|
|
|
NS_ASSERTION (script == this, "That's not my script!");
|
|
|
|
NS_RELEASE(script);
|
2001-06-28 11:46:36 +04:00
|
|
|
|
|
|
|
return NS_OK;
|
|
|
|
}
|
|
|
|
|
|
|
|
NS_IMETHODIMP
|
|
|
|
jsdScript::GetIsValid(PRBool *_rval)
|
|
|
|
{
|
|
|
|
*_rval = mValid;
|
2001-05-02 01:33:43 +04:00
|
|
|
return NS_OK;
|
|
|
|
}
|
|
|
|
|
2001-04-17 12:50:41 +04:00
|
|
|
NS_IMETHODIMP
|
2001-04-20 07:54:05 +04:00
|
|
|
jsdScript::GetFileName(char **_rval)
|
2001-04-17 12:50:41 +04:00
|
|
|
{
|
2001-09-29 12:28:41 +04:00
|
|
|
*_rval = ToNewCString(*mFileName);
|
2001-04-17 12:50:41 +04:00
|
|
|
return NS_OK;
|
|
|
|
}
|
|
|
|
|
|
|
|
NS_IMETHODIMP
|
2001-04-20 07:54:05 +04:00
|
|
|
jsdScript::GetFunctionName(char **_rval)
|
2001-04-17 12:50:41 +04:00
|
|
|
{
|
2001-09-29 12:28:41 +04:00
|
|
|
*_rval = ToNewCString(*mFunctionName);
|
2001-04-17 12:50:41 +04:00
|
|
|
return NS_OK;
|
|
|
|
}
|
|
|
|
|
|
|
|
NS_IMETHODIMP
|
|
|
|
jsdScript::GetBaseLineNumber(PRUint32 *_rval)
|
|
|
|
{
|
2001-06-28 11:46:36 +04:00
|
|
|
*_rval = mBaseLineNumber;
|
2001-04-17 12:50:41 +04:00
|
|
|
return NS_OK;
|
|
|
|
}
|
|
|
|
|
|
|
|
NS_IMETHODIMP
|
|
|
|
jsdScript::GetLineExtent(PRUint32 *_rval)
|
|
|
|
{
|
2001-06-28 11:46:36 +04:00
|
|
|
*_rval = mLineExtent;
|
2001-04-20 07:54:05 +04:00
|
|
|
return NS_OK;
|
|
|
|
}
|
|
|
|
|
2001-04-25 22:20:47 +04:00
|
|
|
NS_IMETHODIMP
|
|
|
|
jsdScript::PcToLine(jsdIPC *aPC, PRUint32 *_rval)
|
|
|
|
{
|
2001-05-02 01:33:43 +04:00
|
|
|
ASSERT_VALID_SCRIPT;
|
2001-04-25 22:20:47 +04:00
|
|
|
jsuword pc;
|
|
|
|
aPC->GetPc(&pc);
|
|
|
|
*_rval = JSD_GetClosestLine (mCx, mScript, pc);
|
|
|
|
return NS_OK;
|
|
|
|
}
|
|
|
|
|
|
|
|
NS_IMETHODIMP
|
|
|
|
jsdScript::LineToPc(PRUint32 aLine, jsdIPC **_rval)
|
|
|
|
{
|
2001-05-02 01:33:43 +04:00
|
|
|
ASSERT_VALID_SCRIPT;
|
2001-04-25 22:20:47 +04:00
|
|
|
jsuword pc = JSD_GetClosestPC (mCx, mScript, aLine);
|
|
|
|
*_rval = jsdPC::FromPtr (pc);
|
|
|
|
return NS_OK;
|
|
|
|
}
|
|
|
|
|
2001-10-30 16:41:32 +03:00
|
|
|
NS_IMETHODIMP
|
|
|
|
jsdScript::IsLineExecutable(PRUint32 aLine, PRBool *_rval)
|
|
|
|
{
|
|
|
|
ASSERT_VALID_SCRIPT;
|
|
|
|
jsuword pc = JSD_GetClosestPC (mCx, mScript, aLine);
|
|
|
|
*_rval = (aLine == JSD_GetClosestLine (mCx, mScript, pc));
|
|
|
|
return NS_OK;
|
|
|
|
}
|
|
|
|
|
2001-04-25 22:20:47 +04:00
|
|
|
NS_IMETHODIMP
|
|
|
|
jsdScript::SetBreakpoint(jsdIPC *aPC)
|
|
|
|
{
|
2001-05-02 01:33:43 +04:00
|
|
|
ASSERT_VALID_SCRIPT;
|
2001-04-25 22:20:47 +04:00
|
|
|
jsuword pc;
|
|
|
|
aPC->GetPc (&pc);
|
|
|
|
|
2001-05-05 02:14:04 +04:00
|
|
|
JSD_SetExecutionHook (mCx, mScript, pc, jsds_ExecutionHookProc,
|
|
|
|
NS_REINTERPRET_CAST(void *, PRIVATE_TO_JSVAL(NULL)));
|
2001-04-25 22:20:47 +04:00
|
|
|
return NS_OK;
|
|
|
|
}
|
|
|
|
|
|
|
|
NS_IMETHODIMP
|
|
|
|
jsdScript::ClearBreakpoint(jsdIPC *aPC)
|
|
|
|
{
|
2001-07-04 02:19:04 +04:00
|
|
|
ASSERT_VALID_SCRIPT;
|
2001-05-14 01:08:06 +04:00
|
|
|
if (!aPC)
|
|
|
|
return NS_ERROR_INVALID_ARG;
|
|
|
|
|
2001-04-25 22:20:47 +04:00
|
|
|
jsuword pc;
|
|
|
|
aPC->GetPc (&pc);
|
|
|
|
|
|
|
|
JSD_ClearExecutionHook (mCx, mScript, pc);
|
|
|
|
return NS_OK;
|
|
|
|
}
|
|
|
|
|
|
|
|
NS_IMETHODIMP
|
|
|
|
jsdScript::ClearAllBreakpoints()
|
|
|
|
{
|
2001-05-02 01:33:43 +04:00
|
|
|
ASSERT_VALID_SCRIPT;
|
2001-05-14 01:08:06 +04:00
|
|
|
JSD_LockScriptSubsystem(mCx);
|
2001-04-25 22:20:47 +04:00
|
|
|
JSD_ClearAllExecutionHooksForScript (mCx, mScript);
|
2001-05-14 01:08:06 +04:00
|
|
|
JSD_UnlockScriptSubsystem(mCx);
|
2001-04-25 22:20:47 +04:00
|
|
|
return NS_OK;
|
|
|
|
}
|
|
|
|
|
2001-04-20 07:54:05 +04:00
|
|
|
/* Stack Frames */
|
2001-07-04 02:19:04 +04:00
|
|
|
NS_IMPL_THREADSAFE_ISUPPORTS2(jsdStackFrame, jsdIStackFrame, jsdIEphemeral);
|
2001-04-20 07:54:05 +04:00
|
|
|
|
|
|
|
NS_IMETHODIMP
|
|
|
|
jsdStackFrame::GetJSDContext(JSDContext **_rval)
|
|
|
|
{
|
2001-07-04 02:19:04 +04:00
|
|
|
ASSERT_VALID_FRAME;
|
2001-04-20 07:54:05 +04:00
|
|
|
*_rval = mCx;
|
|
|
|
return NS_OK;
|
|
|
|
}
|
|
|
|
|
|
|
|
NS_IMETHODIMP
|
|
|
|
jsdStackFrame::GetJSDThreadState(JSDThreadState **_rval)
|
|
|
|
{
|
2001-07-04 02:19:04 +04:00
|
|
|
ASSERT_VALID_FRAME;
|
2001-04-20 07:54:05 +04:00
|
|
|
*_rval = mThreadState;
|
|
|
|
return NS_OK;
|
|
|
|
}
|
|
|
|
|
|
|
|
NS_IMETHODIMP
|
|
|
|
jsdStackFrame::GetJSDStackFrameInfo(JSDStackFrameInfo **_rval)
|
|
|
|
{
|
2001-07-04 02:19:04 +04:00
|
|
|
ASSERT_VALID_FRAME;
|
2001-04-20 07:54:05 +04:00
|
|
|
*_rval = mStackFrameInfo;
|
|
|
|
return NS_OK;
|
|
|
|
}
|
|
|
|
|
2001-07-04 02:19:04 +04:00
|
|
|
NS_IMETHODIMP
|
|
|
|
jsdStackFrame::GetIsValid(PRBool *_rval)
|
|
|
|
{
|
|
|
|
*_rval = mValid;
|
|
|
|
return NS_OK;
|
|
|
|
}
|
|
|
|
|
|
|
|
NS_IMETHODIMP
|
|
|
|
jsdStackFrame::Invalidate()
|
|
|
|
{
|
|
|
|
ASSERT_VALID_FRAME;
|
|
|
|
mValid = PR_FALSE;
|
|
|
|
return NS_OK;
|
|
|
|
}
|
|
|
|
|
2001-04-20 07:54:05 +04:00
|
|
|
NS_IMETHODIMP
|
|
|
|
jsdStackFrame::GetCallingFrame(jsdIStackFrame **_rval)
|
|
|
|
{
|
2001-07-04 02:19:04 +04:00
|
|
|
ASSERT_VALID_FRAME;
|
2001-04-20 07:54:05 +04:00
|
|
|
JSDStackFrameInfo *sfi = JSD_GetCallingStackFrame (mCx, mThreadState,
|
|
|
|
mStackFrameInfo);
|
|
|
|
*_rval = jsdStackFrame::FromPtr (mCx, mThreadState, sfi);
|
|
|
|
return NS_OK;
|
|
|
|
}
|
|
|
|
|
|
|
|
NS_IMETHODIMP
|
|
|
|
jsdStackFrame::GetScript(jsdIScript **_rval)
|
|
|
|
{
|
2001-07-04 02:19:04 +04:00
|
|
|
ASSERT_VALID_FRAME;
|
2001-04-20 07:54:05 +04:00
|
|
|
JSDScript *script = JSD_GetScriptForStackFrame (mCx, mThreadState,
|
|
|
|
mStackFrameInfo);
|
|
|
|
*_rval = jsdScript::FromPtr (mCx, script);
|
|
|
|
return NS_OK;
|
|
|
|
}
|
|
|
|
|
|
|
|
NS_IMETHODIMP
|
2001-04-25 22:20:47 +04:00
|
|
|
jsdStackFrame::GetPc(jsdIPC **_rval)
|
2001-04-20 07:54:05 +04:00
|
|
|
{
|
2001-07-04 02:19:04 +04:00
|
|
|
ASSERT_VALID_FRAME;
|
2001-04-25 22:20:47 +04:00
|
|
|
jsuword pc;
|
|
|
|
pc = JSD_GetPCForStackFrame (mCx, mThreadState, mStackFrameInfo);
|
|
|
|
*_rval = jsdPC::FromPtr (pc);
|
2001-04-17 12:50:41 +04:00
|
|
|
return NS_OK;
|
|
|
|
}
|
|
|
|
|
2001-04-23 23:12:56 +04:00
|
|
|
NS_IMETHODIMP
|
|
|
|
jsdStackFrame::GetLine(PRUint32 *_rval)
|
|
|
|
{
|
2001-07-04 02:19:04 +04:00
|
|
|
ASSERT_VALID_FRAME;
|
2001-04-23 23:12:56 +04:00
|
|
|
JSDScript *script = JSD_GetScriptForStackFrame (mCx, mThreadState,
|
|
|
|
mStackFrameInfo);
|
|
|
|
jsuword pc = JSD_GetPCForStackFrame (mCx, mThreadState, mStackFrameInfo);
|
|
|
|
*_rval = JSD_GetClosestLine (mCx, script, pc);
|
|
|
|
return NS_OK;
|
|
|
|
}
|
|
|
|
|
2001-04-20 07:54:05 +04:00
|
|
|
NS_IMETHODIMP
|
|
|
|
jsdStackFrame::GetCallee(jsdIValue **_rval)
|
|
|
|
{
|
2001-07-04 02:19:04 +04:00
|
|
|
ASSERT_VALID_FRAME;
|
2001-04-20 07:54:05 +04:00
|
|
|
JSDValue *jsdv = JSD_GetCallObjectForStackFrame (mCx, mThreadState,
|
|
|
|
mStackFrameInfo);
|
|
|
|
|
|
|
|
*_rval = jsdValue::FromPtr (mCx, jsdv);
|
|
|
|
return NS_OK;
|
|
|
|
}
|
|
|
|
|
|
|
|
NS_IMETHODIMP
|
|
|
|
jsdStackFrame::GetScope(jsdIValue **_rval)
|
|
|
|
{
|
2001-07-04 02:19:04 +04:00
|
|
|
ASSERT_VALID_FRAME;
|
2001-04-20 07:54:05 +04:00
|
|
|
JSDValue *jsdv = JSD_GetScopeChainForStackFrame (mCx, mThreadState,
|
|
|
|
mStackFrameInfo);
|
|
|
|
|
|
|
|
*_rval = jsdValue::FromPtr (mCx, jsdv);
|
|
|
|
return NS_OK;
|
|
|
|
}
|
|
|
|
|
|
|
|
NS_IMETHODIMP
|
|
|
|
jsdStackFrame::GetThisValue(jsdIValue **_rval)
|
|
|
|
{
|
2001-07-04 02:19:04 +04:00
|
|
|
ASSERT_VALID_FRAME;
|
2001-04-20 07:54:05 +04:00
|
|
|
JSDValue *jsdv = JSD_GetThisForStackFrame (mCx, mThreadState,
|
|
|
|
mStackFrameInfo);
|
|
|
|
|
|
|
|
*_rval = jsdValue::FromPtr (mCx, jsdv);
|
|
|
|
return NS_OK;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
NS_IMETHODIMP
|
|
|
|
jsdStackFrame::Eval (const nsAReadableString &bytes, const char *fileName,
|
|
|
|
PRUint32 line, jsdIValue **_rval)
|
|
|
|
{
|
2001-07-04 02:19:04 +04:00
|
|
|
ASSERT_VALID_FRAME;
|
2001-04-20 07:54:05 +04:00
|
|
|
jsval jv;
|
|
|
|
|
|
|
|
const nsSharedBufferHandle<PRUnichar> *h = bytes.GetSharedBufferHandle();
|
2001-04-21 19:11:02 +04:00
|
|
|
const jschar *char_bytes = NS_REINTERPRET_CAST(const jschar *,
|
|
|
|
h->DataStart());
|
2001-04-20 07:54:05 +04:00
|
|
|
|
|
|
|
if (!JSD_EvaluateUCScriptInStackFrame (mCx, mThreadState, mStackFrameInfo,
|
|
|
|
char_bytes, bytes.Length(), fileName,
|
|
|
|
line, &jv))
|
|
|
|
return NS_ERROR_FAILURE;
|
|
|
|
|
|
|
|
JSDValue *jsdv = JSD_NewValue (mCx, jv);
|
|
|
|
*_rval = jsdValue::FromPtr (mCx, jsdv);
|
|
|
|
return NS_OK;
|
|
|
|
}
|
|
|
|
|
2001-05-05 02:14:04 +04:00
|
|
|
/* Values */
|
2001-07-04 02:19:04 +04:00
|
|
|
NS_IMPL_THREADSAFE_ISUPPORTS2(jsdValue, jsdIValue, jsdIEphemeral);
|
|
|
|
|
|
|
|
jsdValue::jsdValue (JSDContext *aCx, JSDValue *aValue) : mValid(PR_TRUE),
|
|
|
|
mCx(aCx),
|
|
|
|
mValue(aValue)
|
|
|
|
{
|
|
|
|
DEBUG_CREATE ("jsdValue", gValueCount);
|
|
|
|
NS_INIT_ISUPPORTS();
|
|
|
|
mLiveListEntry.value = this;
|
|
|
|
jsds_InsertEphemeral (&gLiveValues, &mLiveListEntry);
|
|
|
|
}
|
|
|
|
|
|
|
|
jsdValue::~jsdValue()
|
|
|
|
{
|
|
|
|
DEBUG_DESTROY ("jsdValue", gValueCount);
|
|
|
|
if (mValid)
|
|
|
|
/* call Invalidate() to take ourselves out of the live list */
|
|
|
|
Invalidate();
|
|
|
|
}
|
|
|
|
|
|
|
|
NS_IMETHODIMP
|
|
|
|
jsdValue::GetIsValid(PRBool *_rval)
|
|
|
|
{
|
|
|
|
*_rval = mValid;
|
|
|
|
return NS_OK;
|
|
|
|
}
|
|
|
|
|
|
|
|
NS_IMETHODIMP
|
|
|
|
jsdValue::Invalidate()
|
|
|
|
{
|
|
|
|
ASSERT_VALID_VALUE;
|
|
|
|
mValid = PR_FALSE;
|
|
|
|
jsds_RemoveEphemeral (&gLiveValues, &mLiveListEntry);
|
|
|
|
JSD_DropValue (mCx, mValue);
|
|
|
|
return NS_OK;
|
|
|
|
}
|
|
|
|
|
|
|
|
void
|
|
|
|
jsdValue::InvalidateAll()
|
|
|
|
{
|
|
|
|
if (gLiveValues)
|
|
|
|
jsds_InvalidateAllEphemerals (&gLiveValues);
|
|
|
|
}
|
2001-04-20 07:54:05 +04:00
|
|
|
|
|
|
|
NS_IMETHODIMP
|
|
|
|
jsdValue::GetJSDContext(JSDContext **_rval)
|
|
|
|
{
|
2001-07-04 02:19:04 +04:00
|
|
|
ASSERT_VALID_VALUE;
|
2001-04-20 07:54:05 +04:00
|
|
|
*_rval = mCx;
|
|
|
|
return NS_OK;
|
|
|
|
}
|
|
|
|
|
|
|
|
NS_IMETHODIMP
|
|
|
|
jsdValue::GetJSDValue (JSDValue **_rval)
|
|
|
|
{
|
2001-07-04 02:19:04 +04:00
|
|
|
ASSERT_VALID_VALUE;
|
2001-04-20 07:54:05 +04:00
|
|
|
*_rval = mValue;
|
|
|
|
return NS_OK;
|
|
|
|
}
|
|
|
|
|
|
|
|
NS_IMETHODIMP
|
|
|
|
jsdValue::GetIsNative (PRBool *_rval)
|
|
|
|
{
|
2001-07-04 02:19:04 +04:00
|
|
|
ASSERT_VALID_VALUE;
|
2001-04-20 07:54:05 +04:00
|
|
|
*_rval = JSD_IsValueNative (mCx, mValue);
|
|
|
|
return NS_OK;
|
|
|
|
}
|
|
|
|
|
|
|
|
NS_IMETHODIMP
|
|
|
|
jsdValue::GetIsNumber (PRBool *_rval)
|
|
|
|
{
|
2001-07-04 02:19:04 +04:00
|
|
|
ASSERT_VALID_VALUE;
|
2001-04-20 07:54:05 +04:00
|
|
|
*_rval = JSD_IsValueNumber (mCx, mValue);
|
|
|
|
return NS_OK;
|
|
|
|
}
|
|
|
|
|
|
|
|
NS_IMETHODIMP
|
|
|
|
jsdValue::GetIsPrimitive (PRBool *_rval)
|
|
|
|
{
|
2001-07-04 02:19:04 +04:00
|
|
|
ASSERT_VALID_VALUE;
|
2001-04-20 07:54:05 +04:00
|
|
|
*_rval = JSD_IsValuePrimitive (mCx, mValue);
|
|
|
|
return NS_OK;
|
|
|
|
}
|
|
|
|
|
|
|
|
NS_IMETHODIMP
|
|
|
|
jsdValue::GetJsType (PRUint32 *_rval)
|
|
|
|
{
|
2001-07-04 02:19:04 +04:00
|
|
|
ASSERT_VALID_VALUE;
|
2001-08-29 02:02:07 +04:00
|
|
|
jsval val;
|
|
|
|
|
|
|
|
val = JSD_GetValueWrappedJSVal (mCx, mValue);
|
|
|
|
|
|
|
|
if (JSVAL_IS_NULL(val))
|
|
|
|
*_rval = TYPE_NULL;
|
|
|
|
else if (JSVAL_IS_BOOLEAN(val))
|
2001-05-05 02:14:04 +04:00
|
|
|
*_rval = TYPE_BOOLEAN;
|
2001-08-29 02:02:07 +04:00
|
|
|
else if (JSVAL_IS_DOUBLE(val))
|
2001-04-20 07:54:05 +04:00
|
|
|
*_rval = TYPE_DOUBLE;
|
2001-08-29 02:02:07 +04:00
|
|
|
else if (JSVAL_IS_INT(val))
|
2001-04-20 07:54:05 +04:00
|
|
|
*_rval = TYPE_INT;
|
2001-08-29 02:02:07 +04:00
|
|
|
else if (JSVAL_IS_STRING(val))
|
2001-04-20 07:54:05 +04:00
|
|
|
*_rval = TYPE_STRING;
|
2001-08-29 02:02:07 +04:00
|
|
|
else if (JSVAL_IS_VOID(val))
|
2001-04-20 07:54:05 +04:00
|
|
|
*_rval = TYPE_VOID;
|
2001-08-29 02:02:07 +04:00
|
|
|
else if (JSD_IsValueFunction (mCx, mValue))
|
|
|
|
*_rval = TYPE_FUNCTION;
|
|
|
|
else if (JSVAL_IS_OBJECT(val))
|
|
|
|
*_rval = TYPE_OBJECT;
|
2001-04-20 07:54:05 +04:00
|
|
|
else
|
2001-08-03 09:14:21 +04:00
|
|
|
NS_ASSERTION (0, "Value has no discernible type.");
|
2001-04-20 07:54:05 +04:00
|
|
|
|
|
|
|
return NS_OK;
|
|
|
|
}
|
|
|
|
|
|
|
|
NS_IMETHODIMP
|
|
|
|
jsdValue::GetJsPrototype (jsdIValue **_rval)
|
|
|
|
{
|
2001-07-04 02:19:04 +04:00
|
|
|
ASSERT_VALID_VALUE;
|
2001-04-20 07:54:05 +04:00
|
|
|
JSDValue *jsdv = JSD_GetValuePrototype (mCx, mValue);
|
|
|
|
*_rval = jsdValue::FromPtr (mCx, jsdv);
|
|
|
|
return NS_OK;
|
|
|
|
}
|
|
|
|
|
|
|
|
NS_IMETHODIMP
|
|
|
|
jsdValue::GetJsParent (jsdIValue **_rval)
|
|
|
|
{
|
2001-07-04 02:19:04 +04:00
|
|
|
ASSERT_VALID_VALUE;
|
2001-04-20 07:54:05 +04:00
|
|
|
JSDValue *jsdv = JSD_GetValueParent (mCx, mValue);
|
|
|
|
*_rval = jsdValue::FromPtr (mCx, jsdv);
|
|
|
|
return NS_OK;
|
|
|
|
}
|
|
|
|
|
|
|
|
NS_IMETHODIMP
|
|
|
|
jsdValue::GetJsClassName(char **_rval)
|
|
|
|
{
|
2001-07-04 02:19:04 +04:00
|
|
|
ASSERT_VALID_VALUE;
|
2001-09-29 12:28:41 +04:00
|
|
|
*_rval = ToNewCString(nsDependentCString(JSD_GetValueClassName(mCx, mValue)));
|
2001-04-20 07:54:05 +04:00
|
|
|
return NS_OK;
|
|
|
|
}
|
|
|
|
|
|
|
|
NS_IMETHODIMP
|
|
|
|
jsdValue::GetJsConstructor (jsdIValue **_rval)
|
|
|
|
{
|
2001-07-04 02:19:04 +04:00
|
|
|
ASSERT_VALID_VALUE;
|
2001-04-20 07:54:05 +04:00
|
|
|
JSDValue *jsdv = JSD_GetValueConstructor (mCx, mValue);
|
|
|
|
*_rval = jsdValue::FromPtr (mCx, jsdv);
|
|
|
|
return NS_OK;
|
|
|
|
}
|
|
|
|
|
|
|
|
NS_IMETHODIMP
|
|
|
|
jsdValue::GetJsFunctionName(char **_rval)
|
|
|
|
{
|
2001-07-04 02:19:04 +04:00
|
|
|
ASSERT_VALID_VALUE;
|
2001-10-30 16:41:32 +03:00
|
|
|
*_rval =
|
|
|
|
ToNewCString(nsDependentCString(JSD_GetValueFunctionName(mCx, mValue)));
|
2001-04-20 07:54:05 +04:00
|
|
|
return NS_OK;
|
|
|
|
}
|
|
|
|
|
|
|
|
NS_IMETHODIMP
|
|
|
|
jsdValue::GetBooleanValue(PRBool *_rval)
|
|
|
|
{
|
2001-07-04 02:19:04 +04:00
|
|
|
ASSERT_VALID_VALUE;
|
2001-04-20 07:54:05 +04:00
|
|
|
*_rval = JSD_GetValueBoolean (mCx, mValue);
|
|
|
|
return NS_OK;
|
|
|
|
}
|
|
|
|
|
|
|
|
NS_IMETHODIMP
|
|
|
|
jsdValue::GetDoubleValue(double *_rval)
|
|
|
|
{
|
2001-07-04 02:19:04 +04:00
|
|
|
ASSERT_VALID_VALUE;
|
2001-08-03 09:14:21 +04:00
|
|
|
double *dp = JSD_GetValueDouble (mCx, mValue);
|
|
|
|
if (dp)
|
|
|
|
*_rval = *dp;
|
|
|
|
else
|
|
|
|
*_rval = 0;
|
2001-04-20 07:54:05 +04:00
|
|
|
return NS_OK;
|
|
|
|
}
|
|
|
|
|
|
|
|
NS_IMETHODIMP
|
|
|
|
jsdValue::GetIntValue(PRInt32 *_rval)
|
|
|
|
{
|
2001-07-04 02:19:04 +04:00
|
|
|
ASSERT_VALID_VALUE;
|
2001-04-20 07:54:05 +04:00
|
|
|
*_rval = JSD_GetValueInt (mCx, mValue);
|
|
|
|
return NS_OK;
|
|
|
|
}
|
|
|
|
|
|
|
|
NS_IMETHODIMP
|
|
|
|
jsdValue::GetObjectValue(jsdIObject **_rval)
|
|
|
|
{
|
2001-07-04 02:19:04 +04:00
|
|
|
ASSERT_VALID_VALUE;
|
2001-04-20 07:54:05 +04:00
|
|
|
JSDObject *obj;
|
|
|
|
obj = JSD_GetObjectForValue (mCx, mValue);
|
|
|
|
*_rval = jsdObject::FromPtr (mCx, obj);
|
|
|
|
return NS_OK;
|
|
|
|
}
|
|
|
|
|
|
|
|
NS_IMETHODIMP
|
|
|
|
jsdValue::GetStringValue(char **_rval)
|
|
|
|
{
|
2001-07-04 02:19:04 +04:00
|
|
|
ASSERT_VALID_VALUE;
|
2001-04-20 07:54:05 +04:00
|
|
|
JSString *jstr_val = JSD_GetValueString(mCx, mValue);
|
2001-09-29 12:28:41 +04:00
|
|
|
*_rval = ToNewCString(nsDependentCString(JS_GetStringBytes(jstr_val)));
|
2001-04-20 07:54:05 +04:00
|
|
|
return NS_OK;
|
|
|
|
}
|
|
|
|
|
2001-04-23 23:12:56 +04:00
|
|
|
NS_IMETHODIMP
|
2001-05-05 02:14:04 +04:00
|
|
|
jsdValue::GetPropertyCount (PRInt32 *_rval)
|
2001-04-23 23:12:56 +04:00
|
|
|
{
|
2001-07-04 02:19:04 +04:00
|
|
|
ASSERT_VALID_VALUE;
|
2001-05-05 02:14:04 +04:00
|
|
|
if (JSD_IsValueObject(mCx, mValue))
|
|
|
|
*_rval = JSD_GetCountOfProperties (mCx, mValue);
|
|
|
|
else
|
|
|
|
*_rval = -1;
|
2001-04-23 23:12:56 +04:00
|
|
|
return NS_OK;
|
|
|
|
}
|
|
|
|
|
2001-04-20 07:54:05 +04:00
|
|
|
NS_IMETHODIMP
|
|
|
|
jsdValue::GetProperties (jsdIProperty ***propArray, PRUint32 *length)
|
|
|
|
{
|
2001-07-04 02:19:04 +04:00
|
|
|
ASSERT_VALID_VALUE;
|
2001-05-05 02:14:04 +04:00
|
|
|
if (!JSD_IsValueObject(mCx, mValue)) {
|
|
|
|
*length = 0;
|
|
|
|
*propArray = 0;
|
|
|
|
return NS_OK;
|
|
|
|
}
|
|
|
|
|
2001-04-21 11:21:27 +04:00
|
|
|
jsdIProperty **pa_temp;
|
|
|
|
PRUint32 prop_count = JSD_GetCountOfProperties (mCx, mValue);
|
|
|
|
|
|
|
|
pa_temp = NS_STATIC_CAST(jsdIProperty **,
|
|
|
|
nsMemory::Alloc(sizeof (jsdIProperty *) *
|
|
|
|
prop_count));
|
|
|
|
|
|
|
|
PRUint32 i = 0;
|
|
|
|
JSDProperty *iter = NULL;
|
|
|
|
JSDProperty *prop;
|
|
|
|
while ((prop = JSD_IterateProperties (mCx, mValue, &iter))) {
|
|
|
|
pa_temp[i] = jsdProperty::FromPtr (mCx, prop);
|
|
|
|
++i;
|
|
|
|
}
|
|
|
|
|
|
|
|
NS_ASSERTION (prop_count == i, "property count mismatch");
|
|
|
|
|
|
|
|
/* if caller doesn't care about length, don't bother telling them */
|
|
|
|
*propArray = pa_temp;
|
|
|
|
if (length)
|
|
|
|
*length = prop_count;
|
|
|
|
|
2001-04-20 07:54:05 +04:00
|
|
|
return NS_OK;
|
|
|
|
}
|
|
|
|
|
|
|
|
NS_IMETHODIMP
|
|
|
|
jsdValue::GetProperty (const char *name, jsdIProperty **_rval)
|
|
|
|
{
|
2001-07-04 02:19:04 +04:00
|
|
|
ASSERT_VALID_VALUE;
|
2001-04-20 07:54:05 +04:00
|
|
|
JSContext *cx = JSD_GetDefaultJSContext (mCx);
|
|
|
|
/* not rooting this */
|
|
|
|
JSString *jstr_name = JS_NewStringCopyZ (cx, name);
|
|
|
|
|
|
|
|
JSDProperty *prop = JSD_GetValueProperty (mCx, mValue, jstr_name);
|
|
|
|
|
|
|
|
*_rval = jsdProperty::FromPtr (mCx, prop);
|
|
|
|
return NS_OK;
|
|
|
|
}
|
|
|
|
|
|
|
|
NS_IMETHODIMP
|
|
|
|
jsdValue::Refresh()
|
|
|
|
{
|
2001-07-04 02:19:04 +04:00
|
|
|
ASSERT_VALID_VALUE;
|
2001-04-20 07:54:05 +04:00
|
|
|
JSD_RefreshValue (mCx, mValue);
|
|
|
|
return NS_OK;
|
|
|
|
}
|
|
|
|
|
2001-04-17 12:50:41 +04:00
|
|
|
/******************************************************************************
|
|
|
|
* debugger service implementation
|
|
|
|
******************************************************************************/
|
|
|
|
NS_IMPL_THREADSAFE_ISUPPORTS1(jsdService, jsdIDebuggerService);
|
|
|
|
|
2001-10-30 16:41:32 +03:00
|
|
|
NS_IMETHODIMP
|
|
|
|
jsdService::GetInitAtStartup (PRBool *_rval)
|
|
|
|
{
|
|
|
|
nsresult rv;
|
|
|
|
nsCOMPtr<nsICategoryManager>
|
|
|
|
categoryManager(do_GetService(NS_CATMAN_CTRID, &rv));
|
|
|
|
if (NS_FAILED(rv))
|
|
|
|
return rv;
|
|
|
|
|
|
|
|
if (mInitAtStartup == triUnknown) {
|
|
|
|
nsXPIDLCString notused;
|
|
|
|
nsresult autoreg_rv, appstart_rv;
|
|
|
|
|
|
|
|
autoreg_rv = categoryManager->GetCategoryEntry(AUTOREG_CATEGORY,
|
|
|
|
JSD_STARTUP_ENTRY,
|
|
|
|
getter_Copies(notused));
|
|
|
|
appstart_rv = categoryManager->GetCategoryEntry(APPSTART_CATEGORY,
|
|
|
|
JSD_STARTUP_ENTRY,
|
|
|
|
getter_Copies(notused));
|
|
|
|
if (autoreg_rv != appstart_rv) {
|
|
|
|
/* we have an inconsistent state in the registry, attempt to fix.
|
|
|
|
* we need to make mInitAtStartup disagree with the state passed
|
|
|
|
* to SetInitAtStartup to make it actually do something.
|
|
|
|
*/
|
|
|
|
mInitAtStartup = triYes;
|
|
|
|
rv = SetInitAtStartup (PR_FALSE);
|
|
|
|
if (NS_FAILED(rv))
|
|
|
|
return rv;
|
|
|
|
} else if (autoreg_rv == NS_ERROR_NOT_AVAILABLE) {
|
|
|
|
mInitAtStartup = triNo;
|
|
|
|
} else if (NS_SUCCEEDED(autoreg_rv)) {
|
|
|
|
mInitAtStartup = triYes;
|
|
|
|
} else {
|
|
|
|
return rv;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if (_rval)
|
|
|
|
*_rval = (mInitAtStartup == triYes);
|
|
|
|
|
|
|
|
return NS_OK;
|
|
|
|
}
|
|
|
|
|
|
|
|
/*
|
|
|
|
* The initAtStartup property controls whether or not we register the
|
|
|
|
* app start observer (jsdASObserver.) We register for both
|
|
|
|
* "xpcom-autoregistration" and "app-startup" notifications if |state| is true.
|
|
|
|
* the autoreg message is sent just before registration occurs (before
|
|
|
|
* "app-startup".) We care about autoreg because it may load javascript
|
|
|
|
* components. autoreg does *not* fire if components haven't changed since the
|
|
|
|
* last autoreg, so we watch "app-startup" as a fallback.
|
|
|
|
*/
|
|
|
|
NS_IMETHODIMP
|
|
|
|
jsdService::SetInitAtStartup (PRBool state)
|
|
|
|
{
|
|
|
|
nsresult rv;
|
|
|
|
|
|
|
|
if (mInitAtStartup == triUnknown) {
|
|
|
|
/* side effect sets mInitAtStartup */
|
|
|
|
rv = GetInitAtStartup(nsnull);
|
|
|
|
if (NS_FAILED(rv))
|
|
|
|
return rv;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (state && mInitAtStartup == triYes ||
|
|
|
|
!state && mInitAtStartup == triNo) {
|
|
|
|
/* already in the requested state */
|
|
|
|
return NS_OK;
|
|
|
|
}
|
|
|
|
|
|
|
|
nsCOMPtr<nsICategoryManager>
|
|
|
|
categoryManager(do_GetService(NS_CATMAN_CTRID, &rv));
|
|
|
|
|
|
|
|
if (state) {
|
|
|
|
rv = categoryManager->AddCategoryEntry(AUTOREG_CATEGORY,
|
|
|
|
JSD_STARTUP_ENTRY,
|
|
|
|
jsdASObserverCtrID,
|
|
|
|
PR_TRUE, PR_TRUE, nsnull);
|
|
|
|
if (NS_FAILED(rv))
|
|
|
|
return rv;
|
|
|
|
rv = categoryManager->AddCategoryEntry(APPSTART_CATEGORY,
|
|
|
|
JSD_STARTUP_ENTRY,
|
|
|
|
jsdASObserverCtrID,
|
|
|
|
PR_TRUE, PR_TRUE, nsnull);
|
|
|
|
if (NS_FAILED(rv))
|
|
|
|
return rv;
|
|
|
|
mInitAtStartup = triYes;
|
|
|
|
} else {
|
|
|
|
rv = categoryManager->DeleteCategoryEntry(AUTOREG_CATEGORY,
|
|
|
|
JSD_STARTUP_ENTRY, PR_TRUE);
|
|
|
|
if (NS_FAILED(rv))
|
|
|
|
return rv;
|
|
|
|
rv = categoryManager->DeleteCategoryEntry(APPSTART_CATEGORY,
|
|
|
|
JSD_STARTUP_ENTRY, PR_TRUE);
|
|
|
|
if (NS_FAILED(rv))
|
|
|
|
return rv;
|
|
|
|
mInitAtStartup = triNo;
|
|
|
|
}
|
|
|
|
|
|
|
|
return NS_OK;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2001-06-28 11:46:36 +04:00
|
|
|
NS_IMETHODIMP
|
|
|
|
jsdService::GetIsOn (PRBool *_rval)
|
2001-04-17 12:50:41 +04:00
|
|
|
{
|
2001-06-28 11:46:36 +04:00
|
|
|
*_rval = mOn;
|
|
|
|
return NS_OK;
|
2001-04-17 12:50:41 +04:00
|
|
|
}
|
|
|
|
|
|
|
|
NS_IMETHODIMP
|
2001-04-25 22:20:47 +04:00
|
|
|
jsdService::On (void)
|
2001-04-17 12:50:41 +04:00
|
|
|
{
|
|
|
|
nsresult rv;
|
|
|
|
|
|
|
|
/* get JS things from the CallContext */
|
|
|
|
nsCOMPtr<nsIXPConnect> xpc = do_GetService(nsIXPConnect::GetCID());
|
|
|
|
if (!xpc) return NS_ERROR_FAILURE;
|
|
|
|
|
|
|
|
nsCOMPtr<nsIXPCNativeCallContext> cc;
|
|
|
|
rv = xpc->GetCurrentNativeCallContext(getter_AddRefs(cc));
|
|
|
|
if (NS_FAILED(rv)) return NS_ERROR_FAILURE;
|
|
|
|
|
|
|
|
JSContext *cx;
|
|
|
|
rv = cc->GetJSContext (&cx);
|
|
|
|
if (NS_FAILED(rv)) return NS_ERROR_FAILURE;
|
|
|
|
|
2001-05-21 10:00:12 +04:00
|
|
|
return OnForRuntime(JS_GetRuntime (cx));
|
|
|
|
|
|
|
|
}
|
2001-04-25 22:20:47 +04:00
|
|
|
|
2001-05-21 10:00:12 +04:00
|
|
|
NS_IMETHODIMP
|
|
|
|
jsdService::OnForRuntime (JSRuntime *rt)
|
|
|
|
{
|
2001-06-28 11:46:36 +04:00
|
|
|
if (mOn)
|
|
|
|
return (rt == mRuntime) ? NS_OK : NS_ERROR_ALREADY_INITIALIZED;
|
|
|
|
|
|
|
|
mRuntime = rt;
|
|
|
|
|
2001-04-26 01:54:59 +04:00
|
|
|
if (gLastGCProc == jsds_GCCallbackProc)
|
|
|
|
/* condition indicates that the callback proc has not been set yet */
|
2001-05-02 01:33:43 +04:00
|
|
|
gLastGCProc = JS_SetGCCallbackRT (rt, jsds_GCCallbackProc);
|
2001-05-21 11:18:10 +04:00
|
|
|
|
2001-06-28 11:46:36 +04:00
|
|
|
mCx = JSD_DebuggerOnForUser (rt, NULL, NULL);
|
2001-04-25 22:20:47 +04:00
|
|
|
if (!mCx)
|
|
|
|
return NS_ERROR_FAILURE;
|
2001-06-28 11:46:36 +04:00
|
|
|
|
2001-09-06 05:34:44 +04:00
|
|
|
JSContext *cx = JSD_GetDefaultJSContext (mCx);
|
|
|
|
JSObject *glob = JS_GetGlobalObject (cx);
|
|
|
|
|
|
|
|
/* init xpconnect on the debugger's context in case xpconnect tries to
|
|
|
|
* use it for stuff. */
|
|
|
|
nsCOMPtr<nsIXPConnect> xpc = do_GetService(nsIXPConnect::GetCID());
|
|
|
|
if (!xpc)
|
|
|
|
return NS_ERROR_FAILURE;
|
|
|
|
|
|
|
|
xpc->InitClasses (cx, glob);
|
|
|
|
|
2001-06-28 11:46:36 +04:00
|
|
|
/* If any of these mFooHook objects are installed, do the required JSD
|
|
|
|
* hookup now. See also, jsdService::SetFooHook().
|
|
|
|
*/
|
|
|
|
if (mThrowHook)
|
|
|
|
JSD_SetThrowHook (mCx, jsds_ExecutionHookProc, NULL);
|
|
|
|
if (mScriptHook)
|
|
|
|
JSD_SetScriptHook (mCx, jsds_ScriptHookProc, NULL);
|
|
|
|
if (mInterruptHook)
|
|
|
|
JSD_SetInterruptHook (mCx, jsds_ExecutionHookProc, NULL);
|
|
|
|
if (mDebuggerHook)
|
|
|
|
JSD_SetDebuggerHook (mCx, jsds_ExecutionHookProc, NULL);
|
|
|
|
if (mErrorHook)
|
|
|
|
JSD_SetDebugBreakHook (mCx, jsds_ExecutionHookProc, NULL);
|
2001-09-01 22:03:53 +04:00
|
|
|
if (mTopLevelHook)
|
|
|
|
JSD_SetTopLevelHook (mCx, jsds_CallHookProc, NULL);
|
|
|
|
else
|
2001-10-30 16:41:32 +03:00
|
|
|
JSD_ClearTopLevelHook (mCx);
|
2001-09-01 22:03:53 +04:00
|
|
|
if (mFunctionHook)
|
|
|
|
JSD_SetFunctionHook (mCx, jsds_CallHookProc, NULL);
|
|
|
|
else
|
2001-10-30 16:41:32 +03:00
|
|
|
JSD_ClearFunctionHook (mCx);
|
2001-04-25 22:20:47 +04:00
|
|
|
mOn = PR_TRUE;
|
2001-05-21 10:00:12 +04:00
|
|
|
|
2001-10-30 16:41:32 +03:00
|
|
|
#ifdef DEBUG
|
|
|
|
printf ("+++ JavaScript debuging hooks installed.\n");
|
|
|
|
#endif
|
2001-04-17 12:50:41 +04:00
|
|
|
return NS_OK;
|
|
|
|
}
|
|
|
|
|
2001-04-25 22:20:47 +04:00
|
|
|
NS_IMETHODIMP
|
|
|
|
jsdService::Off (void)
|
|
|
|
{
|
2001-10-30 16:41:32 +03:00
|
|
|
if (!mOn)
|
|
|
|
return NS_OK;
|
|
|
|
|
2001-06-28 11:46:36 +04:00
|
|
|
if (!mCx || !mRuntime)
|
|
|
|
return NS_ERROR_NOT_INITIALIZED;
|
|
|
|
|
2001-10-30 16:41:32 +03:00
|
|
|
if (gDeadScripts) {
|
2001-06-28 11:46:36 +04:00
|
|
|
if (gGCStatus == JSGC_END)
|
2001-08-29 02:02:07 +04:00
|
|
|
{
|
|
|
|
JSContext *cx = JSD_GetDefaultJSContext(mCx);
|
|
|
|
jsds_NotifyPendingDeadScripts(cx);
|
|
|
|
}
|
2001-06-28 11:46:36 +04:00
|
|
|
else
|
|
|
|
return NS_ERROR_NOT_AVAILABLE;
|
2001-10-30 16:41:32 +03:00
|
|
|
}
|
2001-07-04 02:19:04 +04:00
|
|
|
|
|
|
|
/*
|
2001-06-28 11:46:36 +04:00
|
|
|
if (gLastGCProc != jsds_GCCallbackProc)
|
|
|
|
JS_SetGCCallbackRT (mRuntime, gLastGCProc);
|
2001-07-04 02:19:04 +04:00
|
|
|
*/
|
2001-06-28 11:46:36 +04:00
|
|
|
|
2001-07-04 02:19:04 +04:00
|
|
|
jsdValue::InvalidateAll();
|
|
|
|
jsdProperty::InvalidateAll();
|
2001-06-28 11:46:36 +04:00
|
|
|
ClearAllBreakpoints();
|
|
|
|
|
2001-10-30 16:41:32 +03:00
|
|
|
JSD_ClearThrowHook (mCx);
|
2001-06-28 11:46:36 +04:00
|
|
|
JSD_SetScriptHook (mCx, NULL, NULL);
|
2001-10-30 16:41:32 +03:00
|
|
|
JSD_ClearInterruptHook (mCx);
|
|
|
|
JSD_ClearDebuggerHook (mCx);
|
|
|
|
JSD_ClearDebugBreakHook (mCx);
|
|
|
|
JSD_ClearTopLevelHook (mCx);
|
|
|
|
JSD_ClearFunctionHook (mCx);
|
2001-06-28 11:46:36 +04:00
|
|
|
|
2001-04-25 22:20:47 +04:00
|
|
|
JSD_DebuggerOff (mCx);
|
2001-06-28 11:46:36 +04:00
|
|
|
|
|
|
|
mCx = nsnull;
|
|
|
|
mRuntime = nsnull;
|
2001-04-25 22:20:47 +04:00
|
|
|
mOn = PR_FALSE;
|
2001-10-30 16:41:32 +03:00
|
|
|
|
|
|
|
#ifdef DEBUG
|
|
|
|
printf ("+++ JavaScript debuging hooks removed.\n");
|
|
|
|
#endif
|
|
|
|
|
|
|
|
return NS_OK;
|
|
|
|
}
|
|
|
|
|
|
|
|
NS_IMETHODIMP
|
|
|
|
jsdService::Pause(PRUint32 *_rval)
|
|
|
|
{
|
|
|
|
if (++mPauseLevel == 1) {
|
|
|
|
JSD_ClearThrowHook (mCx);
|
|
|
|
JSD_ClearInterruptHook (mCx);
|
|
|
|
JSD_ClearDebuggerHook (mCx);
|
|
|
|
JSD_ClearDebugBreakHook (mCx);
|
|
|
|
JSD_ClearTopLevelHook (mCx);
|
|
|
|
JSD_ClearFunctionHook (mCx);
|
|
|
|
}
|
|
|
|
|
|
|
|
if (_rval)
|
|
|
|
*_rval = mPauseLevel;
|
|
|
|
|
|
|
|
return NS_OK;
|
|
|
|
}
|
|
|
|
|
|
|
|
NS_IMETHODIMP
|
|
|
|
jsdService::UnPause(PRUint32 *_rval)
|
|
|
|
{
|
|
|
|
if (mPauseLevel == 0)
|
|
|
|
return NS_ERROR_NOT_AVAILABLE;
|
|
|
|
|
|
|
|
if (--mPauseLevel == 0) {
|
|
|
|
if (mThrowHook)
|
|
|
|
JSD_SetThrowHook (mCx, jsds_ExecutionHookProc, NULL);
|
|
|
|
if (mInterruptHook)
|
|
|
|
JSD_SetInterruptHook (mCx, jsds_ExecutionHookProc, NULL);
|
|
|
|
if (mDebuggerHook)
|
|
|
|
JSD_SetDebuggerHook (mCx, jsds_ExecutionHookProc, NULL);
|
|
|
|
if (mErrorHook)
|
|
|
|
JSD_SetDebugBreakHook (mCx, jsds_ExecutionHookProc, NULL);
|
|
|
|
if (mTopLevelHook)
|
|
|
|
JSD_SetTopLevelHook (mCx, jsds_CallHookProc, NULL);
|
|
|
|
else
|
|
|
|
JSD_ClearTopLevelHook (mCx);
|
|
|
|
if (mFunctionHook)
|
|
|
|
JSD_SetFunctionHook (mCx, jsds_CallHookProc, NULL);
|
|
|
|
else
|
|
|
|
JSD_ClearFunctionHook (mCx);
|
|
|
|
}
|
|
|
|
|
|
|
|
if (_rval)
|
|
|
|
*_rval = mPauseLevel;
|
|
|
|
|
2001-04-25 22:20:47 +04:00
|
|
|
return NS_OK;
|
|
|
|
}
|
|
|
|
|
|
|
|
NS_IMETHODIMP
|
2001-05-21 11:18:10 +04:00
|
|
|
jsdService::EnumerateScripts (jsdIScriptEnumerator *enumerator)
|
|
|
|
{
|
2001-06-28 11:46:36 +04:00
|
|
|
ASSERT_VALID_CONTEXT;
|
|
|
|
|
2001-05-21 11:18:10 +04:00
|
|
|
JSDScript *script;
|
|
|
|
JSDScript *iter = NULL;
|
|
|
|
nsresult rv = NS_OK;
|
|
|
|
|
|
|
|
JSD_LockScriptSubsystem(mCx);
|
|
|
|
while((script = JSD_IterateScripts(mCx, &iter)) != NULL) {
|
2001-10-30 16:41:32 +03:00
|
|
|
rv = enumerator->EnumerateScript (jsdScript::FromPtr(mCx, script));
|
|
|
|
if (NS_FAILED(rv))
|
2001-05-21 11:18:10 +04:00
|
|
|
break;
|
|
|
|
}
|
|
|
|
JSD_UnlockScriptSubsystem(mCx);
|
|
|
|
|
|
|
|
return rv;
|
|
|
|
}
|
|
|
|
|
2001-08-03 09:14:21 +04:00
|
|
|
|
|
|
|
NS_IMETHODIMP
|
|
|
|
jsdService::GC (void)
|
|
|
|
{
|
|
|
|
ASSERT_VALID_CONTEXT;
|
|
|
|
JSContext *cx = JSD_GetDefaultJSContext (mCx);
|
|
|
|
JS_GC(cx);
|
|
|
|
return NS_OK;
|
|
|
|
}
|
|
|
|
|
2001-10-30 16:41:32 +03:00
|
|
|
NS_IMETHODIMP
|
|
|
|
jsdService::InsertFilter (jsdIFilter *filter, jsdIFilter *after)
|
|
|
|
{
|
|
|
|
NS_ENSURE_ARG_POINTER (filter);
|
|
|
|
if (jsds_FindFilter (filter))
|
|
|
|
return NS_ERROR_INVALID_ARG;
|
|
|
|
|
|
|
|
FilterRecord *rec = PR_NEWZAP (FilterRecord);
|
|
|
|
|
|
|
|
if (!jsds_SyncFilter (rec, filter)) {
|
|
|
|
PR_Free (rec);
|
|
|
|
return NS_ERROR_FAILURE;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (gFilters) {
|
|
|
|
if (!after) {
|
|
|
|
/* insert at head of list */
|
|
|
|
PR_INSERT_LINK(&rec->links, &gFilters->links);
|
|
|
|
gFilters = rec;
|
|
|
|
} else {
|
|
|
|
/* insert somewhere in the list */
|
|
|
|
FilterRecord *afterRecord = jsds_FindFilter (after);
|
|
|
|
if (!afterRecord) {
|
|
|
|
jsds_FreeFilter(rec);
|
|
|
|
return NS_ERROR_INVALID_ARG;
|
|
|
|
}
|
|
|
|
PR_INSERT_AFTER(&rec->links, &afterRecord->links);
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
if (after) {
|
|
|
|
/* user asked to insert into the middle of an empty list, bail. */
|
|
|
|
jsds_FreeFilter(rec);
|
|
|
|
return NS_ERROR_NOT_INITIALIZED;
|
|
|
|
}
|
|
|
|
PR_INIT_CLIST(&rec->links);
|
|
|
|
gFilters = rec;
|
|
|
|
}
|
|
|
|
|
|
|
|
return NS_OK;
|
|
|
|
}
|
|
|
|
|
|
|
|
NS_IMETHODIMP
|
|
|
|
jsdService::AppendFilter (jsdIFilter *filter)
|
|
|
|
{
|
|
|
|
NS_ENSURE_ARG_POINTER (filter);
|
|
|
|
if (jsds_FindFilter (filter))
|
|
|
|
return NS_ERROR_INVALID_ARG;
|
|
|
|
FilterRecord *rec = PR_NEWZAP (FilterRecord);
|
|
|
|
|
|
|
|
if (!jsds_SyncFilter (rec, filter)) {
|
|
|
|
PR_Free (rec);
|
|
|
|
return NS_ERROR_FAILURE;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (gFilters) {
|
|
|
|
PR_INSERT_BEFORE(&rec->links, &gFilters->links);
|
|
|
|
} else {
|
|
|
|
PR_INIT_CLIST(&rec->links);
|
|
|
|
gFilters = rec;
|
|
|
|
}
|
|
|
|
|
|
|
|
return NS_OK;
|
|
|
|
}
|
|
|
|
|
|
|
|
NS_IMETHODIMP
|
|
|
|
jsdService::RemoveFilter (jsdIFilter *filter)
|
|
|
|
{
|
|
|
|
NS_ENSURE_ARG_POINTER(filter);
|
|
|
|
FilterRecord *rec = jsds_FindFilter (filter);
|
|
|
|
if (!rec)
|
|
|
|
return NS_ERROR_INVALID_ARG;
|
|
|
|
|
|
|
|
if (gFilters == rec)
|
|
|
|
gFilters = NS_REINTERPRET_CAST(FilterRecord *,
|
|
|
|
PR_NEXT_LINK(&rec->links));
|
|
|
|
|
|
|
|
PR_REMOVE_LINK(&rec->links);
|
|
|
|
jsds_FreeFilter (rec);
|
|
|
|
|
|
|
|
return NS_OK;
|
|
|
|
}
|
|
|
|
|
|
|
|
NS_IMETHODIMP
|
|
|
|
jsdService::SwapFilters (jsdIFilter *filter_a, jsdIFilter *filter_b)
|
|
|
|
{
|
|
|
|
NS_ENSURE_ARG_POINTER(filter_a);
|
|
|
|
NS_ENSURE_ARG_POINTER(filter_b);
|
|
|
|
|
|
|
|
FilterRecord *rec_a = jsds_FindFilter (filter_a);
|
|
|
|
if (!rec_a)
|
|
|
|
return NS_ERROR_INVALID_ARG;
|
|
|
|
|
|
|
|
if (filter_a == filter_b) {
|
|
|
|
/* just a refresh */
|
|
|
|
if (!jsds_SyncFilter (rec_a, filter_a))
|
|
|
|
return NS_ERROR_FAILURE;
|
|
|
|
return NS_OK;
|
|
|
|
}
|
|
|
|
|
|
|
|
FilterRecord *rec_b = jsds_FindFilter (filter_b);
|
|
|
|
if (!rec_b) {
|
|
|
|
/* filter_b is not in the list, replace filter_a with filter_b. */
|
|
|
|
if (!jsds_SyncFilter (rec_a, filter_b))
|
|
|
|
return NS_ERROR_FAILURE;
|
|
|
|
} else {
|
|
|
|
/* both filters are in the list, swap. */
|
|
|
|
if (!jsds_SyncFilter (rec_a, filter_b))
|
|
|
|
return NS_ERROR_FAILURE;
|
|
|
|
if (!jsds_SyncFilter (rec_b, filter_a))
|
|
|
|
return NS_ERROR_FAILURE;
|
|
|
|
}
|
|
|
|
|
|
|
|
return NS_OK;
|
|
|
|
}
|
|
|
|
|
|
|
|
NS_IMETHODIMP
|
|
|
|
jsdService::EnumerateFilters (jsdIFilterEnumerator *enumerator)
|
|
|
|
{
|
|
|
|
if (!gFilters)
|
|
|
|
return NS_OK;
|
|
|
|
|
|
|
|
FilterRecord *current = gFilters;
|
|
|
|
do {
|
|
|
|
jsds_SyncFilter (current, current->filterObject);
|
|
|
|
/* SyncFilter failure would be bad, but what would we do about it? */
|
|
|
|
if (enumerator) {
|
|
|
|
nsresult rv = enumerator->EnumerateFilter (current->filterObject);
|
|
|
|
if (NS_FAILED(rv))
|
|
|
|
return rv;
|
|
|
|
}
|
|
|
|
current = NS_REINTERPRET_CAST(FilterRecord *,
|
|
|
|
PR_NEXT_LINK (¤t->links));
|
|
|
|
} while (current != gFilters);
|
|
|
|
|
|
|
|
return NS_OK;
|
|
|
|
}
|
|
|
|
|
|
|
|
NS_IMETHODIMP
|
|
|
|
jsdService::RefreshFilters ()
|
|
|
|
{
|
|
|
|
return EnumerateFilters(nsnull);
|
|
|
|
}
|
|
|
|
|
|
|
|
NS_IMETHODIMP
|
|
|
|
jsdService::ClearFilters ()
|
|
|
|
{
|
|
|
|
if (!gFilters)
|
|
|
|
return NS_OK;
|
|
|
|
|
|
|
|
FilterRecord *current = NS_REINTERPRET_CAST(FilterRecord *,
|
|
|
|
PR_NEXT_LINK (&gFilters->links));
|
|
|
|
do {
|
|
|
|
FilterRecord *next = NS_REINTERPRET_CAST(FilterRecord *,
|
|
|
|
PR_NEXT_LINK (¤t->links));
|
|
|
|
PR_REMOVE_AND_INIT_LINK(¤t->links);
|
|
|
|
jsds_FreeFilter(current);
|
|
|
|
current = next;
|
|
|
|
} while (current != gFilters);
|
|
|
|
|
|
|
|
jsds_FreeFilter(current);
|
|
|
|
gFilters = nsnull;
|
|
|
|
|
|
|
|
return NS_OK;
|
|
|
|
}
|
|
|
|
|
2001-05-21 11:18:10 +04:00
|
|
|
NS_IMETHODIMP
|
2001-04-25 22:20:47 +04:00
|
|
|
jsdService::ClearAllBreakpoints (void)
|
|
|
|
{
|
2001-06-28 11:46:36 +04:00
|
|
|
ASSERT_VALID_CONTEXT;
|
|
|
|
|
2001-05-14 01:08:06 +04:00
|
|
|
JSD_LockScriptSubsystem(mCx);
|
2001-04-25 22:20:47 +04:00
|
|
|
JSD_ClearAllExecutionHooks (mCx);
|
2001-05-14 01:08:06 +04:00
|
|
|
JSD_UnlockScriptSubsystem(mCx);
|
2001-04-25 22:20:47 +04:00
|
|
|
return NS_OK;
|
|
|
|
}
|
|
|
|
|
2001-04-20 07:54:05 +04:00
|
|
|
NS_IMETHODIMP
|
2001-10-10 10:35:46 +04:00
|
|
|
jsdService::EnterNestedEventLoop (jsdINestCallback *callback, PRUint32 *_rval)
|
2001-04-20 07:54:05 +04:00
|
|
|
{
|
|
|
|
nsCOMPtr<nsIAppShell> appShell(do_CreateInstance(kAppShellCID));
|
|
|
|
NS_ENSURE_TRUE(appShell, NS_ERROR_FAILURE);
|
2001-09-06 01:29:59 +04:00
|
|
|
nsCOMPtr<nsIEventQueueService>
|
|
|
|
eventService(do_GetService(kEventQueueServiceCID));
|
|
|
|
NS_ENSURE_TRUE(eventService, NS_ERROR_FAILURE);
|
|
|
|
|
2001-04-20 07:54:05 +04:00
|
|
|
appShell->Create(0, nsnull);
|
|
|
|
appShell->Spinup();
|
|
|
|
|
|
|
|
nsCOMPtr<nsIJSContextStack>
|
|
|
|
stack(do_GetService("@mozilla.org/js/xpc/ContextStack;1"));
|
|
|
|
nsresult rv = NS_OK;
|
|
|
|
PRUint32 nestLevel = ++mNestedLoopLevel;
|
|
|
|
|
2001-09-06 01:29:59 +04:00
|
|
|
nsCOMPtr<nsIEventQueue> eventQ;
|
|
|
|
if (stack && NS_SUCCEEDED(stack->Push(nsnull)) &&
|
|
|
|
NS_SUCCEEDED(eventService->PushThreadEventQueue(getter_AddRefs(eventQ))))
|
2001-04-20 07:54:05 +04:00
|
|
|
{
|
2001-10-30 16:41:32 +03:00
|
|
|
if (NS_SUCCEEDED(rv) && callback) {
|
|
|
|
Pause(nsnull);
|
2001-10-10 10:35:46 +04:00
|
|
|
rv = callback->OnNest();
|
2001-10-30 16:41:32 +03:00
|
|
|
UnPause(nsnull);
|
2001-10-10 10:35:46 +04:00
|
|
|
}
|
|
|
|
|
2001-04-20 07:54:05 +04:00
|
|
|
while(NS_SUCCEEDED(rv) && mNestedLoopLevel >= nestLevel)
|
|
|
|
{
|
|
|
|
void* data;
|
|
|
|
PRBool isRealEvent;
|
|
|
|
//PRBool processEvent;
|
|
|
|
|
|
|
|
rv = appShell->GetNativeEvent(isRealEvent, data);
|
|
|
|
if(NS_SUCCEEDED(rv))
|
|
|
|
appShell->DispatchNativeEvent(isRealEvent, data);
|
|
|
|
}
|
|
|
|
JSContext* cx;
|
|
|
|
stack->Pop(&cx);
|
|
|
|
NS_ASSERTION(cx == nsnull, "JSContextStack mismatch");
|
|
|
|
}
|
|
|
|
else
|
|
|
|
rv = NS_ERROR_FAILURE;
|
|
|
|
|
2001-09-06 01:29:59 +04:00
|
|
|
eventService->PopThreadEventQueue(eventQ);
|
2001-04-20 07:54:05 +04:00
|
|
|
appShell->Spindown();
|
|
|
|
|
|
|
|
NS_ASSERTION (mNestedLoopLevel <= nestLevel,
|
|
|
|
"nested event didn't unwind properly");
|
|
|
|
if (mNestedLoopLevel == nestLevel)
|
|
|
|
--mNestedLoopLevel;
|
|
|
|
|
|
|
|
*_rval = mNestedLoopLevel;
|
|
|
|
return rv;
|
|
|
|
}
|
|
|
|
|
|
|
|
NS_IMETHODIMP
|
|
|
|
jsdService::ExitNestedEventLoop (PRUint32 *_rval)
|
|
|
|
{
|
|
|
|
if (mNestedLoopLevel > 0)
|
|
|
|
--mNestedLoopLevel;
|
|
|
|
else
|
|
|
|
return NS_ERROR_FAILURE;
|
|
|
|
|
|
|
|
*_rval = mNestedLoopLevel;
|
|
|
|
return NS_OK;
|
|
|
|
}
|
|
|
|
|
2001-04-17 12:50:41 +04:00
|
|
|
/* hook attribute get/set functions */
|
|
|
|
|
2001-04-25 22:20:47 +04:00
|
|
|
NS_IMETHODIMP
|
|
|
|
jsdService::SetBreakpointHook (jsdIExecutionHook *aHook)
|
|
|
|
{
|
|
|
|
mBreakpointHook = aHook;
|
|
|
|
return NS_OK;
|
|
|
|
}
|
|
|
|
|
|
|
|
NS_IMETHODIMP
|
|
|
|
jsdService::GetBreakpointHook (jsdIExecutionHook **aHook)
|
|
|
|
{
|
|
|
|
*aHook = mBreakpointHook;
|
|
|
|
NS_IF_ADDREF(*aHook);
|
|
|
|
|
|
|
|
return NS_OK;
|
|
|
|
}
|
|
|
|
|
2001-04-17 12:50:41 +04:00
|
|
|
NS_IMETHODIMP
|
2001-05-05 02:14:04 +04:00
|
|
|
jsdService::SetErrorHook (jsdIExecutionHook *aHook)
|
2001-04-17 12:50:41 +04:00
|
|
|
{
|
2001-05-05 02:14:04 +04:00
|
|
|
mErrorHook = aHook;
|
2001-06-28 11:46:36 +04:00
|
|
|
|
|
|
|
/* if the debugger isn't initialized, that's all we can do for now. The
|
|
|
|
* OnForRuntime() method will do the rest when the coast is clear.
|
|
|
|
*/
|
2001-10-30 16:41:32 +03:00
|
|
|
if (!mCx || mPauseLevel)
|
2001-06-28 11:46:36 +04:00
|
|
|
return NS_OK;
|
|
|
|
|
2001-04-17 12:50:41 +04:00
|
|
|
if (aHook)
|
2001-05-05 02:14:04 +04:00
|
|
|
JSD_SetDebugBreakHook (mCx, jsds_ExecutionHookProc, NULL);
|
2001-04-17 12:50:41 +04:00
|
|
|
else
|
2001-04-25 22:20:47 +04:00
|
|
|
JSD_ClearDebugBreakHook (mCx);
|
2001-04-17 12:50:41 +04:00
|
|
|
|
|
|
|
return NS_OK;
|
|
|
|
}
|
|
|
|
|
|
|
|
NS_IMETHODIMP
|
2001-05-05 02:14:04 +04:00
|
|
|
jsdService::GetErrorHook (jsdIExecutionHook **aHook)
|
2001-04-17 12:50:41 +04:00
|
|
|
{
|
2001-05-05 02:14:04 +04:00
|
|
|
*aHook = mErrorHook;
|
2001-04-17 12:50:41 +04:00
|
|
|
NS_IF_ADDREF(*aHook);
|
|
|
|
|
|
|
|
return NS_OK;
|
|
|
|
}
|
|
|
|
|
|
|
|
NS_IMETHODIMP
|
|
|
|
jsdService::SetDebuggerHook (jsdIExecutionHook *aHook)
|
|
|
|
{
|
|
|
|
mDebuggerHook = aHook;
|
2001-06-28 11:46:36 +04:00
|
|
|
|
|
|
|
/* if the debugger isn't initialized, that's all we can do for now. The
|
|
|
|
* OnForRuntime() method will do the rest when the coast is clear.
|
|
|
|
*/
|
2001-10-30 16:41:32 +03:00
|
|
|
if (!mCx || mPauseLevel)
|
2001-06-28 11:46:36 +04:00
|
|
|
return NS_OK;
|
|
|
|
|
2001-04-17 12:50:41 +04:00
|
|
|
if (aHook)
|
2001-05-05 02:14:04 +04:00
|
|
|
JSD_SetDebuggerHook (mCx, jsds_ExecutionHookProc, NULL);
|
2001-04-17 12:50:41 +04:00
|
|
|
else
|
2001-04-25 22:20:47 +04:00
|
|
|
JSD_ClearDebuggerHook (mCx);
|
2001-04-17 12:50:41 +04:00
|
|
|
|
|
|
|
return NS_OK;
|
|
|
|
}
|
|
|
|
|
|
|
|
NS_IMETHODIMP
|
|
|
|
jsdService::GetDebuggerHook (jsdIExecutionHook **aHook)
|
|
|
|
{
|
|
|
|
*aHook = mDebuggerHook;
|
|
|
|
NS_IF_ADDREF(*aHook);
|
|
|
|
|
|
|
|
return NS_OK;
|
|
|
|
}
|
|
|
|
|
|
|
|
NS_IMETHODIMP
|
|
|
|
jsdService::SetInterruptHook (jsdIExecutionHook *aHook)
|
|
|
|
{
|
|
|
|
mInterruptHook = aHook;
|
2001-06-28 11:46:36 +04:00
|
|
|
|
|
|
|
/* if the debugger isn't initialized, that's all we can do for now. The
|
|
|
|
* OnForRuntime() method will do the rest when the coast is clear.
|
|
|
|
*/
|
2001-10-30 16:41:32 +03:00
|
|
|
if (!mCx || mPauseLevel)
|
2001-06-28 11:46:36 +04:00
|
|
|
return NS_OK;
|
|
|
|
|
2001-04-17 12:50:41 +04:00
|
|
|
if (aHook)
|
2001-05-05 02:14:04 +04:00
|
|
|
JSD_SetInterruptHook (mCx, jsds_ExecutionHookProc, NULL);
|
2001-04-17 12:50:41 +04:00
|
|
|
else
|
2001-04-25 22:20:47 +04:00
|
|
|
JSD_ClearInterruptHook (mCx);
|
2001-04-17 12:50:41 +04:00
|
|
|
|
|
|
|
return NS_OK;
|
|
|
|
}
|
|
|
|
|
|
|
|
NS_IMETHODIMP
|
|
|
|
jsdService::GetInterruptHook (jsdIExecutionHook **aHook)
|
|
|
|
{
|
|
|
|
*aHook = mInterruptHook;
|
|
|
|
NS_IF_ADDREF(*aHook);
|
|
|
|
|
|
|
|
return NS_OK;
|
|
|
|
}
|
|
|
|
|
|
|
|
NS_IMETHODIMP
|
|
|
|
jsdService::SetScriptHook (jsdIScriptHook *aHook)
|
|
|
|
{
|
|
|
|
mScriptHook = aHook;
|
2001-06-28 11:46:36 +04:00
|
|
|
|
|
|
|
/* if the debugger isn't initialized, that's all we can do for now. The
|
|
|
|
* OnForRuntime() method will do the rest when the coast is clear.
|
|
|
|
*/
|
2001-10-30 16:41:32 +03:00
|
|
|
if (!mCx || mPauseLevel)
|
2001-06-28 11:46:36 +04:00
|
|
|
return NS_OK;
|
|
|
|
|
2001-04-21 11:21:27 +04:00
|
|
|
if (aHook)
|
2001-05-02 01:33:43 +04:00
|
|
|
JSD_SetScriptHook (mCx, jsds_ScriptHookProc, NULL);
|
2001-04-21 11:21:27 +04:00
|
|
|
else
|
2001-04-25 22:20:47 +04:00
|
|
|
JSD_SetScriptHook (mCx, NULL, NULL);
|
2001-04-21 11:21:27 +04:00
|
|
|
|
2001-04-17 12:50:41 +04:00
|
|
|
return NS_OK;
|
|
|
|
}
|
|
|
|
|
|
|
|
NS_IMETHODIMP
|
|
|
|
jsdService::GetScriptHook (jsdIScriptHook **aHook)
|
|
|
|
{
|
|
|
|
*aHook = mScriptHook;
|
|
|
|
NS_IF_ADDREF(*aHook);
|
|
|
|
|
|
|
|
return NS_OK;
|
|
|
|
}
|
|
|
|
|
2001-05-05 02:14:04 +04:00
|
|
|
NS_IMETHODIMP
|
|
|
|
jsdService::SetThrowHook (jsdIExecutionHook *aHook)
|
|
|
|
{
|
|
|
|
mThrowHook = aHook;
|
2001-06-28 11:46:36 +04:00
|
|
|
|
|
|
|
/* if the debugger isn't initialized, that's all we can do for now. The
|
|
|
|
* OnForRuntime() method will do the rest when the coast is clear.
|
|
|
|
*/
|
2001-10-30 16:41:32 +03:00
|
|
|
if (!mCx || mPauseLevel)
|
2001-06-28 11:46:36 +04:00
|
|
|
return NS_OK;
|
|
|
|
|
2001-05-05 02:14:04 +04:00
|
|
|
if (aHook)
|
|
|
|
JSD_SetThrowHook (mCx, jsds_ExecutionHookProc, NULL);
|
|
|
|
else
|
|
|
|
JSD_ClearThrowHook (mCx);
|
|
|
|
|
|
|
|
return NS_OK;
|
|
|
|
}
|
|
|
|
|
|
|
|
NS_IMETHODIMP
|
|
|
|
jsdService::GetThrowHook (jsdIExecutionHook **aHook)
|
|
|
|
{
|
|
|
|
*aHook = mThrowHook;
|
|
|
|
NS_IF_ADDREF(*aHook);
|
|
|
|
|
|
|
|
return NS_OK;
|
|
|
|
}
|
|
|
|
|
2001-09-01 22:03:53 +04:00
|
|
|
NS_IMETHODIMP
|
|
|
|
jsdService::SetTopLevelHook (jsdICallHook *aHook)
|
|
|
|
{
|
|
|
|
mTopLevelHook = aHook;
|
|
|
|
|
|
|
|
/* if the debugger isn't initialized, that's all we can do for now. The
|
|
|
|
* OnForRuntime() method will do the rest when the coast is clear.
|
|
|
|
*/
|
2001-10-30 16:41:32 +03:00
|
|
|
if (!mCx || mPauseLevel)
|
2001-09-01 22:03:53 +04:00
|
|
|
return NS_OK;
|
|
|
|
|
|
|
|
if (aHook)
|
|
|
|
JSD_SetTopLevelHook (mCx, jsds_CallHookProc, NULL);
|
|
|
|
else
|
2001-10-30 16:41:32 +03:00
|
|
|
JSD_ClearTopLevelHook (mCx);
|
2001-09-01 22:03:53 +04:00
|
|
|
|
|
|
|
return NS_OK;
|
|
|
|
}
|
|
|
|
|
|
|
|
NS_IMETHODIMP
|
|
|
|
jsdService::GetTopLevelHook (jsdICallHook **aHook)
|
|
|
|
{
|
|
|
|
*aHook = mTopLevelHook;
|
|
|
|
NS_IF_ADDREF(*aHook);
|
|
|
|
|
|
|
|
return NS_OK;
|
|
|
|
}
|
|
|
|
|
|
|
|
NS_IMETHODIMP
|
|
|
|
jsdService::SetFunctionHook (jsdICallHook *aHook)
|
|
|
|
{
|
|
|
|
mFunctionHook = aHook;
|
|
|
|
|
|
|
|
/* if the debugger isn't initialized, that's all we can do for now. The
|
|
|
|
* OnForRuntime() method will do the rest when the coast is clear.
|
|
|
|
*/
|
2001-10-30 16:41:32 +03:00
|
|
|
if (!mCx || mPauseLevel)
|
2001-09-01 22:03:53 +04:00
|
|
|
return NS_OK;
|
|
|
|
|
|
|
|
if (aHook)
|
|
|
|
JSD_SetFunctionHook (mCx, jsds_CallHookProc, NULL);
|
|
|
|
else
|
2001-10-30 16:41:32 +03:00
|
|
|
JSD_ClearFunctionHook (mCx);
|
2001-09-01 22:03:53 +04:00
|
|
|
|
|
|
|
return NS_OK;
|
|
|
|
}
|
|
|
|
|
|
|
|
NS_IMETHODIMP
|
|
|
|
jsdService::GetFunctionHook (jsdICallHook **aHook)
|
|
|
|
{
|
|
|
|
*aHook = mFunctionHook;
|
|
|
|
NS_IF_ADDREF(*aHook);
|
|
|
|
|
|
|
|
return NS_OK;
|
|
|
|
}
|
|
|
|
|
2001-04-26 01:54:59 +04:00
|
|
|
jsdService *
|
|
|
|
jsdService::GetService ()
|
|
|
|
{
|
|
|
|
if (!gJsds)
|
|
|
|
gJsds = new jsdService();
|
|
|
|
|
|
|
|
NS_IF_ADDREF(gJsds);
|
|
|
|
return gJsds;
|
|
|
|
}
|
|
|
|
|
|
|
|
NS_GENERIC_FACTORY_SINGLETON_CONSTRUCTOR(jsdService, jsdService::GetService);
|
2001-04-17 12:50:41 +04:00
|
|
|
|
2001-10-30 16:41:32 +03:00
|
|
|
/* app-start observer. turns on the debugger at app-start. this is inserted
|
|
|
|
* and/or removed from the app-start category by the jsdService::initAtStartup
|
|
|
|
* property.
|
|
|
|
*/
|
|
|
|
class jsdASObserver : public nsIObserver
|
|
|
|
{
|
|
|
|
public:
|
|
|
|
NS_DECL_ISUPPORTS
|
|
|
|
NS_DECL_NSIOBSERVER
|
|
|
|
|
|
|
|
jsdASObserver ()
|
|
|
|
{
|
|
|
|
NS_INIT_ISUPPORTS();
|
|
|
|
}
|
|
|
|
};
|
|
|
|
|
|
|
|
NS_IMPL_THREADSAFE_ISUPPORTS1(jsdASObserver, nsIObserver);
|
|
|
|
|
|
|
|
NS_IMETHODIMP
|
|
|
|
jsdASObserver::Observe (nsISupports *aSubject, const char *aTopic,
|
|
|
|
const PRUnichar *aData)
|
|
|
|
{
|
|
|
|
nsresult rv;
|
|
|
|
|
|
|
|
jsdService *jsds = jsdService::GetService();
|
|
|
|
PRBool on;
|
|
|
|
rv = jsds->GetIsOn(&on);
|
|
|
|
if (NS_FAILED(rv) || on)
|
|
|
|
return rv;
|
|
|
|
|
|
|
|
nsCOMPtr<nsIJSRuntimeService> rts = do_GetService(NS_JSRT_CTRID, &rv);
|
|
|
|
if (NS_FAILED(rv))
|
|
|
|
return rv;
|
|
|
|
|
|
|
|
JSRuntime *rt;
|
|
|
|
rts->GetRuntime (&rt);
|
|
|
|
if (NS_FAILED(rv))
|
|
|
|
return rv;
|
|
|
|
|
|
|
|
rv = jsds->OnForRuntime(rt);
|
|
|
|
|
|
|
|
return rv;
|
|
|
|
}
|
|
|
|
|
|
|
|
NS_GENERIC_FACTORY_CONSTRUCTOR(jsdASObserver);
|
|
|
|
|
2001-04-17 12:50:41 +04:00
|
|
|
static nsModuleComponentInfo components[] = {
|
2001-10-30 16:41:32 +03:00
|
|
|
{"JSDService", JSDSERVICE_CID, jsdServiceCtrID, jsdServiceConstructor},
|
|
|
|
{"JSDASObserver", JSDASO_CID, jsdASObserverCtrID, jsdASObserverConstructor}
|
2001-04-17 12:50:41 +04:00
|
|
|
};
|
|
|
|
|
2001-06-21 00:21:49 +04:00
|
|
|
NS_IMPL_NSGETMODULE(JavaScript_Debugger, components);
|
2001-05-05 02:14:04 +04:00
|
|
|
|
2001-05-21 10:00:12 +04:00
|
|
|
/********************************************************************************
|
|
|
|
********************************************************************************
|
|
|
|
* graveyard
|
|
|
|
*/
|
2001-05-05 02:14:04 +04:00
|
|
|
|
|
|
|
#if 0
|
|
|
|
/* Thread States */
|
|
|
|
NS_IMPL_THREADSAFE_ISUPPORTS1(jsdThreadState, jsdIThreadState);
|
|
|
|
|
|
|
|
NS_IMETHODIMP
|
|
|
|
jsdThreadState::GetJSDContext(JSDContext **_rval)
|
|
|
|
{
|
|
|
|
*_rval = mCx;
|
|
|
|
return NS_OK;
|
|
|
|
}
|
|
|
|
|
|
|
|
NS_IMETHODIMP
|
|
|
|
jsdThreadState::GetJSDThreadState(JSDThreadState **_rval)
|
|
|
|
{
|
|
|
|
*_rval = mThreadState;
|
|
|
|
return NS_OK;
|
|
|
|
}
|
|
|
|
|
|
|
|
NS_IMETHODIMP
|
|
|
|
jsdThreadState::GetFrameCount (PRUint32 *_rval)
|
|
|
|
{
|
|
|
|
*_rval = JSD_GetCountOfStackFrames (mCx, mThreadState);
|
|
|
|
return NS_OK;
|
|
|
|
}
|
|
|
|
|
|
|
|
NS_IMETHODIMP
|
|
|
|
jsdThreadState::GetTopFrame (jsdIStackFrame **_rval)
|
|
|
|
{
|
|
|
|
JSDStackFrameInfo *sfi = JSD_GetStackFrame (mCx, mThreadState);
|
|
|
|
|
|
|
|
*_rval = jsdStackFrame::FromPtr (mCx, mThreadState, sfi);
|
|
|
|
return NS_OK;
|
|
|
|
}
|
|
|
|
|
|
|
|
NS_IMETHODIMP
|
|
|
|
jsdThreadState::GetPendingException(jsdIValue **_rval)
|
|
|
|
{
|
|
|
|
JSDValue *jsdv = JSD_GetException (mCx, mThreadState);
|
|
|
|
|
|
|
|
*_rval = jsdValue::FromPtr (mCx, jsdv);
|
|
|
|
return NS_OK;
|
|
|
|
}
|
|
|
|
|
|
|
|
NS_IMETHODIMP
|
|
|
|
jsdThreadState::SetPendingException(jsdIValue *aException)
|
|
|
|
{
|
|
|
|
JSDValue *jsdv;
|
|
|
|
|
|
|
|
nsresult rv = aException->GetJSDValue (&jsdv);
|
|
|
|
if (NS_FAILED(rv))
|
|
|
|
return NS_ERROR_FAILURE;
|
|
|
|
|
|
|
|
if (!JSD_SetException (mCx, mThreadState, jsdv))
|
|
|
|
return NS_ERROR_FAILURE;
|
|
|
|
|
|
|
|
return NS_OK;
|
|
|
|
}
|
|
|
|
|
|
|
|
#endif
|