/* -*- Mode: C; tab-width: 8; indent-tabs-mode: nil; c-basic-offset: 4 -*- * * ***** BEGIN LICENSE BLOCK ***** * Version: MPL 1.1/GPL 2.0/LGPL 2.1 * * 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 the Initial Developer are Copyright (C) 1998 * the Initial Developer. All Rights Reserved. * * Contributor(s): * * Alternatively, the contents of this file may be used under the terms of * either the GNU General Public License Version 2 or later (the "GPL"), or * the GNU Lesser General Public License Version 2.1 or later (the "LGPL"), * in which case the provisions of the GPL or the LGPL are applicable instead * of those above. If you wish to allow use of your version of this file only * under the terms of either the GPL or the LGPL, and not to allow others to * use your version of this file under the terms of the MPL, indicate your * decision by deleting the provisions above and replace them with the notice * and other provisions required by the GPL or the LGPL. If you do not delete * the provisions above, a recipient may use your version of this file under * the terms of any one of the MPL, the GPL or the LGPL. * * ***** END LICENSE BLOCK ***** */ /* * jsd_iiop.cpp * * iiop interface to allow access to server backend from client frontend. * * John Bandhauer (jband@netscape.com) * 8/06/97 * */ #ifndef SERVER_BUILD #error jsd_iiop.cpp is for LiveWire only #endif #include "jsdebug.h" #include "jsapi.h" #include "jsstr.h" #include "prhash.h" #include "prmon.h" #include "prlog.h" /* for PR_ASSERT */ #include "prevent.h" #include #include #include #include #include #include "corba.h" #include "NameLib.hpp" #include "ifaces_s.hh" // this shouldn't have to be here, but is not defined elsewhere CosNaming::Name::~Name() {} // #define JSD_IIOP_TRACE 1 #ifdef JSD_IIOP_TRACE // quick hack... static void _write_to_logfile(const char* str) { static const char logfilename[] = "c:\\temp\\jsdiiop.log"; static FILE* logfile = NULL; static PRMonitor* mon = NULL; if( NULL == mon && NULL == (mon = PR_NewMonitor()) ) return; PR_EnterMonitor(mon); if( NULL == logfile && NULL == (logfile = fopen(logfilename, "a+")) ) { PR_ExitMonitor(mon); return; } #ifdef WIN32 int threadid = (int) GetCurrentThreadId(); #else int threadid = (int) PR_GetCurrentThread(); #endif char* buf = PR_smprintf("thread: \"%0x\" msg: \"%s\"\n", threadid, str); fwrite(buf, sizeof(char), strlen(buf), logfile); fflush(logfile); free(buf); PR_ExitMonitor(mon); } // #define TRACE(msg) log_error(LOG_VERBOSE, "jsd_iiop", NULL, NULL, "TRACE: %s", msg) #define TRACE(msg) _write_to_logfile(msg) #else #define TRACE(msg) #endif /***************************************************************************/ // deal with the fact that char* return values to CORBA implemented methods // do not get freed! static void _deferedFree(void* ptr) { // it is important that this number be high enough -- no recursive or // huge thread count usage of this function! #define DEFERED_FREE_PTR_COUNT 50 static void* cache[DEFERED_FREE_PTR_COUNT]; static current = 0; static PRMonitor* mon = NULL; if( NULL == mon ) { // init the monitor mon = PR_NewMonitor(); if( NULL == mon ) return; PR_EnterMonitor(mon); // belt and suspenders :) for( int i = 0; i < DEFERED_FREE_PTR_COUNT; i++ ) cache[i] = NULL; } else PR_EnterMonitor(mon); if( cache[current] ) free( cache[current] ); cache[current] = ptr; current++ ; if( current == DEFERED_FREE_PTR_COUNT ) current = 0; PR_ExitMonitor(mon); } /***************************************************************************/ // // local declarations // // prototype these for proper extern 'C' PR_BEGIN_EXTERN_C PR_EXTERN(int) ssjsdebugInit(pblock *pb, Session *sn, Request *rq); PR_EXTERN(PRUintn) ssjsdErrorReporter( JSDContext* jsdc, JSContext* cx, const char* message, JSErrorReport* report, void* callerdata ); PR_EXTERN(void) ssjsdScriptHookProc( JSDContext* jsdc, JSDScript* jsdscript, JSBool creating, void* callerdata ); PR_EXTERN(PRUintn) ssjsdExecutionHookProc( JSDContext* jsdc, JSDThreadState* jsdthreadstate, PRUintn type, void* callerdata ); PR_END_EXTERN_C /**************************/ static PRMonitor* _jsd_iiop_monitor = NULL; #define LOCK() \ PR_BEGIN_MACRO \ if( ! _jsd_iiop_monitor ) \ _jsd_iiop_monitor = PR_NewMonitor(); \ PR_ASSERT(_jsd_iiop_monitor); \ PR_EnterMonitor(_jsd_iiop_monitor); \ PR_END_MACRO #define UNLOCK() \ PR_BEGIN_MACRO \ PR_ASSERT(_jsd_iiop_monitor); \ PR_ExitMonitor(_jsd_iiop_monitor); \ PR_END_MACRO /**************************/ class TestInterface_impl : public _sk_TestInterface { public: TestInterface_impl() : _sk_TestInterface("bogus") {} virtual char* getFirstAppInList(); virtual void getAppNames(StringReciever_ptr sr); virtual TestInterface::sequence_of_Thing * getThings(); virtual void callBounce(StringReciever_ptr arg0, CORBA::Long arg1); }; /**************************/ class SourceTextProvider : public _sk_ISourceTextProvider { public: SourceTextProvider(const char* name); SourceTextProvider(const char* name, JSDContext* jsdc); virtual ~SourceTextProvider(); void init(JSDContext* jsdc); // implement ISourceTextProvider virtual ISourceTextProvider::sequence_of_string * getAllPages(); virtual void refreshAllPages(); virtual CORBA::Boolean hasPage(const char * arg0); virtual CORBA::Boolean loadPage(const char * arg0); virtual void refreshPage(const char * arg0); virtual char * getPageText(const char * arg0); virtual CORBA::Long getPageStatus(const char * arg0); virtual CORBA::Long getPageAlterCount(const char * arg0); protected: void _forcePreLoad(); void _lock() {JSD_LockSourceTextSubsystem(_jsdc);} void _unlock() {JSD_UnlockSourceTextSubsystem(_jsdc);} private: JSDContext* _jsdc; }; /**************************/ class HookKey { public: HookKey(const IJSPC& pc); IJSPC pc; PRHashNumber hash_num; }; /**************************/ // forward declaration... class StepHandler; /**************************/ class DebugController : public _sk_IDebugController { public: DebugController(const char* name); DebugController(const char* name, JSDContext* jsdc); virtual ~DebugController(); void init(JSDContext* jsdc); // implement IDebugController virtual CORBA::Long getMajorVersion() {return JSD_GetMajorVersion();} virtual CORBA::Long getMinorVersion() {return JSD_GetMinorVersion();} virtual IJSErrorReporter_ptr setErrorReporter(IJSErrorReporter_ptr arg0); virtual IJSErrorReporter_ptr getErrorReporter(); virtual IScriptHook_ptr setScriptHook(IScriptHook_ptr arg0); virtual IScriptHook_ptr getScriptHook(); virtual IJSPC * getClosestPC(const IScript& arg0, CORBA::Long arg1); virtual IJSSourceLocation * getSourceLocation(const IJSPC& arg0); virtual IJSExecutionHook_ptr setInterruptHook(IJSExecutionHook_ptr arg0); virtual IJSExecutionHook_ptr getInterruptHook(); virtual IJSExecutionHook_ptr setDebugBreakHook(IJSExecutionHook_ptr arg0); virtual IJSExecutionHook_ptr getDebugBreakHook(); virtual IJSExecutionHook_ptr setInstructionHook(IJSExecutionHook_ptr arg0, const IJSPC& arg1); virtual IJSExecutionHook_ptr getInstructionHook(const IJSPC& arg0); virtual CORBA::Boolean isRunningHook(CORBA::Long arg0); virtual CORBA::Boolean isWaitingForResume(CORBA::Long arg0); virtual void setThreadContinueState(CORBA::Long arg0, CORBA::Long arg1); virtual void setThreadReturnValue(CORBA::Long arg0, const char * arg1); virtual void sendInterrupt(); virtual void sendInterruptStepInto(CORBA::Long arg0); virtual void sendInterruptStepOver(CORBA::Long arg0); virtual void sendInterruptStepOut(CORBA::Long arg0); virtual void reinstateStepper(CORBA::Long arg0); virtual IExecResult * executeScriptInStackFrame(CORBA::Long arg0, const IJSStackFrameInfo& arg1, const char * arg2, const char * arg3, CORBA::Long arg4); virtual void leaveThreadSuspended(CORBA::Long arg0); virtual void resumeThread(CORBA::Long arg0); virtual void iterateScripts(IScriptHook_ptr arg0); public: // these are called by extern 'C' callbacks (otherwise would be private) int _callErrorReporter(const char* msg, JSErrorReport* report); void _callScriptHook(JSDScript* jsdscript, JSBool creating); int _callExecutionHook(JSDThreadState* jsdthreadstate, PRUintn type); private: void _initScript(IScript& s, CORBA::Long jsdscript, PRBool alive); inline void _initPC(IJSPC& pc, const IScript& script, CORBA::Long offset) { pc.script = script; pc.offset = offset; } inline void _initSourceLocation(IJSSourceLocation& sl, const IJSPC& pc, CORBA::Long line) { sl.pc = pc; sl.line = line; } inline void _initStackFrameInfo(IJSStackFrameInfo& sf, const IJSPC& pc, CORBA::Long jsdframe) { sf.pc = pc; sf.jsdframe = jsdframe; } IJSExecutionHook_ptr _findInstructionHook(const IJSPC& pc); PRBool _addInstructionHook(const IJSPC& pc, IJSExecutionHook_ptr hook); IJSExecutionHook_ptr _removeInstructionHook(const IJSPC& pc); PRBool _initInstructionHookTable(); void _freeInstructionHookTable(); inline void _lock() {PR_EnterMonitor(_monitor);} inline void _unlock() {PR_ExitMonitor(_monitor);} JSDScript* _findScriptAt(const char* filename, int lineno); void _clearStepHandler(PRBool canReinstate); void _reinstateStepHandler(); // data... private: JSDContext* _jsdc; IJSErrorReporter_ptr _errorReporter; IScriptHook_ptr _scriptHook; IJSExecutionHook_ptr _interruptHook; IJSExecutionHook_ptr _debugBreakHook; PRHashTable * _instructionHookTable; PRMonitor* _monitor; StepHandler* _stepHandler; StepHandler* _oldStepHandler; }; /**************************/ class ThreadState { private: // only our static can construct one of us ThreadState(); // not implemented ThreadState(PRThread* currentThread); ~ThreadState(); public: static ThreadState* getThreadStateForCurrentThread(void); static ThreadState* findThreadStateForCurrentThread(void); void callHook(DebugController* controller, IJSExecutionHook* hook, sequence_of_IJSStackFrameInfo& stack, IJSPC* pcTop, JSDThreadState* jsdthreadstate); PRBool isRunningHook(void); PRBool isWaitingForResume(void); void leaveThreadSuspended(void); void resumeThread(); inline PREventQueue* getEventQueue(void) {return _mainEventQueue;} inline IJSThreadState& getJSThreadState(void) {return _jsts;} // these (especially set) should be called on this thread's ThreadState PRBool getRunningEval(void) {return _isRunningEval;} void setRunningEval(PRBool b) {_isRunningEval = b;} IExecResult& getExecResult(void) {return _execResult;} public: // these should be called by the appropriate Events only... void eventSaysSuicide(void); void eventSaysResume(void); void eventSaysCallHook(void); private: // private statics... inline static void _lock() { if(! _monitor) _monitor = PR_NewMonitor(); PR_ASSERT(_monitor); PR_EnterMonitor(_monitor); } inline static void _unlock() { PR_ASSERT(_monitor); PR_ExitMonitor(_monitor); } inline static void _wait() { PR_ASSERT(_monitor); PR_Wait(_monitor, PR_INTERVAL_NO_TIMEOUT); } inline static void _notify() { PR_ASSERT(_monitor); PR_Notify(_monitor); } static void _otherThreadProc(void* arg); static void _callbackForPRThreadDeath(void *arg); static PRHashNumber _hash_root(const void *key); static PRBool _prepThread2IndexTable(); static PRBool _findIndexForThread(PRThread* thread, PRUintn* index); static PRBool _setIndexForThread(PRThread* thread, PRUintn index); static PRBool _removeIndexForThread(PRThread* thread); static ThreadState* _findThreadState(PRThread* currentThread); static ThreadState* _createThreadState(PRThread* currentThread); private: // private instance methods PRBool _initEventQueues(void); private: static PRMonitor* _monitor; static PRHashTable* _thread2IndexTable; PRThread* _currentThread; DebugController* _controller; IJSExecutionHook* _hook; IJSThreadState _jsts; IJSPC* _pcTop; PRBool _isRunningEval; IExecResult _execResult; PRBool _isRunningHook; PRBool _actualCallToHookIsHappening; PRBool _leaveSuspended; PREventQueue* _mainEventQueue; PREventQueue* _hookCallerEventQueue; PRBool _hookCallerEventQueueCreateFinished; PRBool _suicideMessageReceived; }; /***************************************************************************/ // These Event classes all MUST cast to PREvent. So.... // 1) no virtual functions // 2) PREvent must be first in struct OR class must inherit from PREvent class JSDResumeEvent : public PREvent { public: JSDResumeEvent(ThreadState* ts); void post(PREventQueue* queue); static void* static_handle(PREvent* e); static void static_destroy(PREvent* e); private: ThreadState* _ts; }; /**************************/ class JSDSuicideEvent : public PREvent { public: JSDSuicideEvent(ThreadState* ts); void post(PREventQueue* queue); static void* static_handle(PREvent* e); static void static_destroy(PREvent* e); private: ThreadState* _ts; }; /**************************/ class JSDCallHookEvent : public PREvent { public: JSDCallHookEvent(ThreadState* ts); void post(PREventQueue* queue); static void* static_handle(PREvent* e); static void static_destroy(PREvent* e); private: ThreadState* _ts; }; /**************************/ class JSDExecuteEvent : public PREvent { public: JSDExecuteEvent(JSDContext* jsdc, ThreadState* ts, const IJSStackFrameInfo& frame, const char* text, const char* filename, int lineno); ~JSDExecuteEvent(); IExecResult* postSynchronous(PREventQueue* queue); static void* static_handle(PREvent* e); static void static_destroy(PREvent* e); private: void* _handle(); private: JSDContext* _jsdc; ThreadState* _ts; IJSStackFrameInfo _frame; char* _text; char* _filename; int _lineno; }; /**************************/ // support for stepping... class CallChain { public: enum CompareResult { EQUAL, CALLER, CALLEE, DISJOINT }; CallChain(); // not implemented CallChain(JSDContext* jsdc, JSDThreadState* jsdthreadstate); ~CallChain(); CallChain::CompareResult compare(const CallChain& other); // data... private: JSDContext* _jsdc; JSDScript** _chain; int _count; }; /************************/ class StepHandler { public: enum StepResult { STOP, CONTINUE_SEND_INTERRUPT, CONTINUE_DONE }; StepHandler(); // not implemented StepHandler(JSDContext* jsdc, ThreadState* state, PRBool buildCallChain); ~StepHandler(); virtual StepHandler::StepResult step(JSDThreadState* jsdthreadstate) = 0; protected: JSDStackFrameInfo* _topFrame(JSDThreadState* jsdthreadstate) { return JSD_GetStackFrame(_jsdc, jsdthreadstate); } JSDScript* _topScript(JSDThreadState* jsdthreadstate, JSDStackFrameInfo* jsdframe) { return JSD_GetScriptForStackFrame(_jsdc, jsdthreadstate, jsdframe); } prword_t _topPC(JSDThreadState* jsdthreadstate, JSDStackFrameInfo* jsdframe) { return JSD_GetPCForStackFrame(_jsdc, jsdthreadstate, jsdframe); } int _topLine(JSDScript* jsdscript, prword_t pc) { return JSD_GetClosestLine(_jsdc, jsdscript, pc); } // data... protected: JSDContext* _jsdc; JSDScript* topScriptInitial; prword_t topPCInitial; int topLineInitial; CallChain* _callChain; }; /************************/ class StepInto : public StepHandler { public: StepInto(JSDContext* jsdc, ThreadState* state); StepHandler::StepResult step(JSDThreadState* jsdthreadstate); private: }; /************************/ class StepOver : public StepHandler { public: StepOver(JSDContext* jsdc, ThreadState* state); StepHandler::StepResult step(JSDThreadState* jsdthreadstate); private: }; /************************/ class StepOut : public StepHandler { public: StepOut(JSDContext* jsdc, ThreadState* state); StepHandler::StepResult step(JSDThreadState* jsdthreadstate); private: }; /***************************************************************************/ /***************************************************************************/ PR_IMPLEMENT(PRUintn) ssjsdErrorReporter( JSDContext* jsdc, JSContext* cx, const char* message, JSErrorReport* report, void* callerdata ) { DebugController* controller = (DebugController*) callerdata; return (PRUintn) controller->_callErrorReporter(message, report); } PR_IMPLEMENT(void) ssjsdScriptHookProc( JSDContext* jsdc, JSDScript* jsdscript, JSBool creating, void* callerdata ) { DebugController* controller = (DebugController*) callerdata; controller->_callScriptHook(jsdscript, creating); } PR_IMPLEMENT(PRUintn) ssjsdExecutionHookProc( JSDContext* jsdc, JSDThreadState* jsdthreadstate, PRUintn type, void* callerdata ) { DebugController* controller = (DebugController*) callerdata; return (PRUintn) controller->_callExecutionHook(jsdthreadstate, type); } DebugController::DebugController(const char* name, JSDContext* jsdc) : _sk_IDebugController(name), _jsdc(NULL), _errorReporter(NULL), _scriptHook(NULL), _interruptHook(NULL), _debugBreakHook(NULL), _instructionHookTable(NULL), _stepHandler(NULL), _oldStepHandler(NULL) { init(jsdc); } DebugController::DebugController(const char* name) : _sk_IDebugController(name), _jsdc(NULL), _errorReporter(NULL), _scriptHook(NULL), _interruptHook(NULL), _debugBreakHook(NULL), _instructionHookTable(NULL), _stepHandler(NULL), _oldStepHandler(NULL) { // do nothing } void DebugController::init(JSDContext* jsdc) { _jsdc = jsdc; _monitor = PR_NewMonitor(); JSD_SetErrorReporter(_jsdc, ssjsdErrorReporter, this); JSD_SetScriptHook(_jsdc, ssjsdScriptHookProc, this); JSD_SetDebugBreakHook(_jsdc, ssjsdExecutionHookProc, this ); _initInstructionHookTable(); } DebugController::~DebugController() { if( _jsdc ) { JSD_SetErrorReporter(_jsdc, NULL, NULL); JSD_SetScriptHook(_jsdc, NULL, NULL); JSD_SetDebugBreakHook(_jsdc, NULL, NULL ); _freeInstructionHookTable(); } if(_monitor) PR_DestroyMonitor(_monitor); } IJSErrorReporter_ptr DebugController::setErrorReporter(IJSErrorReporter_ptr arg0) { TRACE("DebugController::setErrorReporter called"); IJSErrorReporter_ptr old = _errorReporter; _errorReporter = arg0; if( _errorReporter ) _errorReporter->_duplicate(_errorReporter); // no need to release the old hook, because returning it here // will cause a release call. return old; } IJSErrorReporter_ptr DebugController::getErrorReporter() { TRACE("DebugController::getErrorReporter called"); if( _errorReporter ) return _errorReporter->_duplicate(_errorReporter); return NULL; } IScriptHook_ptr DebugController::setScriptHook(IScriptHook_ptr arg0) { TRACE("DebugController::setScriptHook called"); IScriptHook_ptr old = _scriptHook; _scriptHook = arg0; if( _scriptHook ) _scriptHook->_duplicate(_scriptHook); // no need to release the old hook, because returning it here // will cause a release call. return old; } IScriptHook_ptr DebugController::getScriptHook() { TRACE("DebugController::getScriptHook called"); if( _scriptHook ) return _scriptHook->_duplicate(_scriptHook); return NULL; } IJSExecutionHook_ptr DebugController::setInterruptHook(IJSExecutionHook_ptr arg0) { TRACE("DebugController::setInterruptHook called"); IJSExecutionHook_ptr old = _interruptHook; _interruptHook = arg0; if( _interruptHook ) _interruptHook->_duplicate(_interruptHook); // no need to release the old hook, because returning it here // will cause a release call. return old; } IJSExecutionHook_ptr DebugController::getInterruptHook() { TRACE("DebugController::getInterruptHook called"); if( _interruptHook ) return _interruptHook->_duplicate(_interruptHook); return NULL; } IJSExecutionHook_ptr DebugController::setDebugBreakHook(IJSExecutionHook_ptr arg0) { TRACE("DebugController::setDebugBreakHook called"); IJSExecutionHook_ptr old = _debugBreakHook; _debugBreakHook = arg0; if( _debugBreakHook ) _debugBreakHook->_duplicate(_debugBreakHook); // no need to release the old hook, because returning it here // will cause a release call. return old; } IJSExecutionHook_ptr DebugController::getDebugBreakHook() { TRACE("DebugController::getDebugBreakHook called"); if( _debugBreakHook ) return _debugBreakHook->_duplicate(_debugBreakHook); return NULL; } IJSExecutionHook_ptr DebugController::setInstructionHook(IJSExecutionHook_ptr arg0, const IJSPC& arg1) { TRACE("DebugController::setInstructionHook called"); IJSExecutionHook_ptr oldHook = _findInstructionHook(arg1); if( oldHook ) _removeInstructionHook(arg1); if(arg0) { arg0->_duplicate(arg0); _addInstructionHook(arg1, arg0); } // no need to release the old hook, because returning it here // will cause a release call. return oldHook; } IJSExecutionHook_ptr DebugController::getInstructionHook(const IJSPC& arg0) { TRACE("DebugController::getInstructionHook() called"); IJSExecutionHook_ptr hook = _findInstructionHook(arg0); if( hook ) return hook->_duplicate(hook); return NULL; } IJSPC* DebugController::getClosestPC(const IScript& arg0, CORBA::Long arg1) { TRACE("DebugController::getClosestPC() called"); PRUintn offset = JSD_GetClosestPC(_jsdc, (JSDScript*)arg0.jsdscript, arg1); IJSPC* pc = new IJSPC(); _initPC(*pc, arg0, (CORBA::Long) offset); return pc; } IJSSourceLocation * DebugController::getSourceLocation(const IJSPC& arg0) { TRACE("DebugController::getSourceLocation() called"); JSDScript* jsdscript = (JSDScript*)arg0.script.jsdscript; int line = JSD_GetClosestLine(_jsdc, jsdscript, (PRUintn) arg0.offset); int offset = JSD_GetClosestPC(_jsdc, jsdscript, line); IJSPC pc; _initPC(pc, arg0.script, (CORBA::Long) offset); IJSSourceLocation* loc = new IJSSourceLocation(); _initSourceLocation(*loc, pc, (CORBA::Long)line); return loc; } void DebugController::setThreadContinueState(CORBA::Long arg0, CORBA::Long arg1) { TRACE("DebugController::setThreadContinueState() called"); ThreadState* state = (ThreadState*)arg0; state->getJSThreadState().continueState = arg1; } void DebugController::setThreadReturnValue(CORBA::Long arg0, const char * arg1) { TRACE("DebugController::setThreadReturnValue() called"); ThreadState* state = (ThreadState*)arg0; state->getJSThreadState().returnValue = arg1; } void DebugController::sendInterrupt() { TRACE("DebugController::sendInterrupt() called"); _clearStepHandler(PR_FALSE); JSD_SetInterruptHook(_jsdc, ssjsdExecutionHookProc, this ); } void DebugController::sendInterruptStepInto(CORBA::Long arg0) { TRACE("DebugController::sendInterruptStepInto() called"); _clearStepHandler(PR_FALSE); _stepHandler = new StepInto(_jsdc, (ThreadState*) arg0); JSD_SetInterruptHook(_jsdc, ssjsdExecutionHookProc, this ); } void DebugController::sendInterruptStepOver(CORBA::Long arg0) { TRACE("DebugController::sendInterruptStepOver() called"); _clearStepHandler(PR_FALSE); _stepHandler = new StepOver(_jsdc, (ThreadState*) arg0); JSD_SetInterruptHook(_jsdc, ssjsdExecutionHookProc, this ); } void DebugController::sendInterruptStepOut(CORBA::Long arg0) { TRACE("DebugController::sendInterruptStepOut() called"); _clearStepHandler(PR_FALSE); _stepHandler = new StepOut(_jsdc, (ThreadState*) arg0); JSD_SetInterruptHook(_jsdc, ssjsdExecutionHookProc, this ); } void DebugController::reinstateStepper(CORBA::Long arg0) { TRACE("DebugController::reinstateStepper() called"); _reinstateStepHandler(); } IExecResult * DebugController::executeScriptInStackFrame(CORBA::Long arg0, const IJSStackFrameInfo& arg1, const char * arg2, const char * arg3, CORBA::Long arg4) { TRACE("DebugController::executeScriptInStackFrame() called"); ThreadState* state = (ThreadState*) arg0; if( ! state->isRunningHook() ) { PR_ASSERT(0); return 0; } JSDExecuteEvent* e = new JSDExecuteEvent(_jsdc,state,arg1,arg2,arg3,arg4); return e->postSynchronous(state->getEventQueue()); } CORBA::Boolean DebugController::isRunningHook(CORBA::Long arg0) { TRACE("DebugController::isRunningHook() called"); ThreadState* state = (ThreadState*) arg0; return state->isRunningHook(); } CORBA::Boolean DebugController::isWaitingForResume(CORBA::Long arg0) { TRACE("DebugController::isWaitingForResume() called"); ThreadState* state = (ThreadState*) arg0; return state->isWaitingForResume(); } void DebugController::leaveThreadSuspended(CORBA::Long arg0) { TRACE("DebugController::leaveThreadSuspended() called"); ThreadState* state = (ThreadState*) arg0; state->leaveThreadSuspended(); } void DebugController::resumeThread(CORBA::Long arg0) { TRACE("DebugController::resumeThread() called"); ThreadState* state = (ThreadState*) arg0; state->resumeThread(); } void DebugController::iterateScripts(IScriptHook_ptr arg0) { IScript script; JSDScript* jsdscript; JSDScript* iter = NULL; JSD_LockScriptSubsystem(_jsdc); while( NULL != (jsdscript = JSD_IterateScripts(_jsdc, &iter)) ) { _initScript(script, (long) jsdscript, PR_TRUE); try { TRACE("+++HOOK arg0->justLoadedScript() call hook"); arg0->justLoadedScript(script); TRACE("---HOOK arg0->justLoadedScript() call hook"); } catch(...) { // eat it... } } JSD_UnlockScriptSubsystem(_jsdc); } /**************************************/ void DebugController::_initScript(IScript& s, CORBA::Long jsdscript, PRBool alive) { s.url = JSD_GetScriptFilename(_jsdc, (JSDScript*)jsdscript); s.funname = JSD_GetScriptFunctionName(_jsdc, (JSDScript*)jsdscript); s.base = JSD_GetScriptBaseLineNumber(_jsdc, (JSDScript*)jsdscript); s.extent = JSD_GetScriptLineExtent(_jsdc, (JSDScript*)jsdscript); s.jsdscript = jsdscript; // init the sections... LWDBGScript* lwscript = JSDLW_GetLWScript(_jsdc, (JSDScript*)jsdscript); PRUintn count = 0; // iterate once to get the count... // BUT: don't iterate if the script is being destroyed... if( alive ) { PRUintn iter = 0; while( LWDBG_ScriptSectionIterator(lwscript,NULL,NULL,NULL,&iter) ) count++; } // if sections exist, then grab those with 'user' code PRBool addedAtLeastOneSection = PR_FALSE; if( count ) { IScriptSection* sections = new IScriptSection[count]; int index = 0; PRUintn base = s.base; PRUintn extent; PRUintn type; PRUintn iter = 0; while( LWDBG_ScriptSectionIterator(lwscript,&type,&extent,NULL,&iter) ) { PRBool hasCode; switch( type ) { case LWDBG_SECTIONTYPE_OTHER: hasCode = PR_FALSE; break; case LWDBG_SECTIONTYPE_SERVER: case LWDBG_SECTIONTYPE_BACKQUOTE: case LWDBG_SECTIONTYPE_EQBACKQUOTE: hasCode = PR_TRUE; break; case LWDBG_SECTIONTYPE_URL: default: hasCode = PR_FALSE; break; } if( hasCode ) { sections[index].base = base; sections[index].extent = extent+1; // XXX hackage... index++; } base += extent; } if( index ) { s.sections = sequence_of_IScriptSection(count,index,sections,1); addedAtLeastOneSection = PR_TRUE; } else { delete [] sections; } } if( ! addedAtLeastOneSection ) { PRUintn baseBefore = s.base; if(alive) JSDLW_ProcessedToRawLineNumber(_jsdc, (JSDScript*)jsdscript, s.base, &baseBefore ); // no sections added, add the whole thing as a single section IScriptSection* sections = new IScriptSection[1]; sections[0].base = baseBefore; sections[0].extent = s.extent; s.sections = sequence_of_IScriptSection(1,1,sections,1); } } int DebugController::_callErrorReporter(const char* msg, JSErrorReport* report) { if( ! _errorReporter ) return JSD_ERROR_REPORTER_PASS_ALONG; const char* filename = NULL; unsigned int lineno = 0; const char* linebuf = NULL; int tokenOffset = 0; int retval = JSD_ERROR_REPORTER_PASS_ALONG; if( report ) { filename = report->filename; linebuf = report->linebuf; lineno = report->lineno; if( report->linebuf && report->tokenptr ) tokenOffset = report->tokenptr - report->linebuf; // convert linenumber to Raw JSDScript* jsdscript = _findScriptAt(filename, lineno); if( jsdscript ) JSDLW_ProcessedToRawLineNumber(_jsdc, jsdscript, lineno, &lineno); } // if we are doing an eval on this thread, then gather the info from the // report and then tell JSD to ignore it. ThreadState* ts = ThreadState::findThreadStateForCurrentThread(); if( ts && ts->getRunningEval() ) { IExecResult& result = ts->getExecResult(); result.errorOccured = 1; result.errorMessage = (const char*) msg; result.errorFilename = (const char*) filename; result.errorLineNumber = lineno; result.errorLineBuffer = (const char*) linebuf; result.errorTokenOffset = tokenOffset; return JSD_ERROR_REPORTER_RETURN; } // else.... try { TRACE("+++HOOK _errorReporter->reportError() call hook"); retval = _errorReporter->reportError(msg,filename,lineno,linebuf,tokenOffset); TRACE("---HOOK _errorReporter->reportError() call hook"); } catch(...) { // eat it... } return retval; } void DebugController::_callScriptHook(JSDScript* jsdscript, JSBool creating) { if( ! _scriptHook ) return; // The debugger does not need to know about scripts created and destroyed // while doing eval on that thread (which is what must be happening if // we have the thread stopped, yet have reaced this hook) // // This seemed to be causing deadlocks in calling back to the debugger // under some circumstances ThreadState* ts = ThreadState::findThreadStateForCurrentThread(); if( ts && ts->isRunningHook() ) return; IScript script; _initScript(script, (long) jsdscript, creating); try { if( creating ) { TRACE("+++HOOK _scriptHook->justLoadedScript() call hook"); _scriptHook->justLoadedScript(script); TRACE("---HOOK _scriptHook->justLoadedScript() call hook"); } else { TRACE("+++HOOK _scriptHook->aboutToUnloadScript() call hook"); _scriptHook->aboutToUnloadScript(script); TRACE("---HOOK _scriptHook->aboutToUnloadScript() call hook"); } } catch(...) { // eat it... } } int DebugController::_callExecutionHook(JSDThreadState* jsdthreadstate, PRUintn type) { IJSExecutionHook_ptr hook = NULL; JSDStackFrameInfo* jsdframeTop = JSD_GetStackFrame(_jsdc, jsdthreadstate); IScript scriptTop; _initScript(scriptTop, (CORBA::Long) JSD_GetScriptForStackFrame(_jsdc, jsdthreadstate,jsdframeTop), PR_TRUE ); IJSPC pcTop; _initPC(pcTop, scriptTop, (CORBA::Long) JSD_GetPCForStackFrame(_jsdc, jsdthreadstate, jsdframeTop)); switch(type) { case JSD_HOOK_INTERRUPTED: hook = _interruptHook; if( _stepHandler ) { switch( _stepHandler->step(jsdthreadstate) ) { case StepHandler::STOP: _clearStepHandler(PR_FALSE); break; case StepHandler::CONTINUE_SEND_INTERRUPT: hook = NULL; break; case StepHandler::CONTINUE_DONE: _clearStepHandler(PR_FALSE); hook = NULL; break; default: PR_ASSERT(0); } } else { JSD_ClearInterruptHook(_jsdc); } break; case JSD_HOOK_BREAKPOINT: _clearStepHandler(PR_TRUE); JSD_ClearInterruptHook(_jsdc); hook = _findInstructionHook(pcTop); break; case JSD_HOOK_DEBUG_REQUESTED: _clearStepHandler(PR_FALSE); JSD_ClearInterruptHook(_jsdc); hook = _debugBreakHook; break; default: // invalid type! PR_ASSERT(0); break; } if( ! hook ) return JSD_HOOK_RETURN_CONTINUE; // generate the stack (reverse the order) int count = JSD_GetCountOfStackFrames(_jsdc, jsdthreadstate); IJSStackFrameInfo* frames = new IJSStackFrameInfo[count]; JSDStackFrameInfo* jsdframe = JSD_GetStackFrame(_jsdc, jsdthreadstate); int i = count; while( jsdframe ) { IScript script; _initScript(script, (CORBA::Long) JSD_GetScriptForStackFrame(_jsdc, jsdthreadstate,jsdframe), PR_TRUE); IJSPC pc; _initPC(pc, script, (CORBA::Long) JSD_GetPCForStackFrame(_jsdc, jsdthreadstate, jsdframe)); _initStackFrameInfo(frames[--i], pc, (CORBA::Long) jsdframe); jsdframe = JSD_GetCallingStackFrame(_jsdc, jsdthreadstate, jsdframe); } PR_ASSERT(0==i); sequence_of_IJSStackFrameInfo stack(count,count,frames,1); ThreadState* state = ThreadState::getThreadStateForCurrentThread(); if( ! state ) { PR_ASSERT(0); return JSD_HOOK_RETURN_CONTINUE; } if( state->isRunningHook() ) { TRACE("hit a hook when this thread was already stopped. continuing..." ); return JSD_HOOK_RETURN_CONTINUE; } state->callHook(this, hook, stack, &pcTop, jsdthreadstate); return state->getJSThreadState().continueState; } JSDScript* DebugController::_findScriptAt(const char* filename, int lineno) { JSDScript* script; JSDScript* best_script = NULL; JSDScript* iter = NULL; while( NULL != (script = JSD_IterateScripts(_jsdc, &iter)) ) { if( 0 == strcmp( filename, JSD_GetScriptFilename(_jsdc,script) ) ) { int first = JSD_GetScriptBaseLineNumber(_jsdc, script); int last = first + JSD_GetScriptLineExtent(_jsdc, script) - 1; if( lineno >= first && lineno <= last ) { // save this as best best_script = script; // if this is a function, then we're done. if( NULL != JSD_GetScriptFunctionName(_jsdc, script) ) break; } } } return best_script; } /**************************/ PR_STATIC_CALLBACK(PRHashNumber) _hash_root(const void *key) { return ((HookKey*)key)->hash_num; } PR_STATIC_CALLBACK(PRIntn) _hash_key_comparer(const void *v1, const void *v2) { if( ( ((HookKey*)v1)->pc.script.jsdscript == ((HookKey*)v2)->pc.script.jsdscript ) && ( ((HookKey*)v1)->pc.offset == ((HookKey*)v2)->pc.offset ) ) return 1; return 0; } PR_STATIC_CALLBACK(PRIntn) _hash_entry_zapper(PRHashEntry *he, PRIntn i, void *arg) { // XXX clear the JSD_HOOK ??? delete ((HookKey*)he->key); ((IJSExecutionHook_ptr)he->value)->_release(); he->value = NULL; he->key = NULL; return HT_ENUMERATE_NEXT; } IJSExecutionHook_ptr DebugController::_findInstructionHook(const IJSPC& pc) { IJSExecutionHook_ptr hook = NULL; if( _instructionHookTable ) { _lock(); HookKey key(pc); hook = (IJSExecutionHook_ptr) PR_HashTableLookup(_instructionHookTable, &key); _unlock(); } return hook; } PRBool DebugController::_addInstructionHook(const IJSPC& pc, IJSExecutionHook_ptr hook) { PRBool retval = PR_FALSE; if( _instructionHookTable ) { HookKey* key = new HookKey(pc); _lock(); retval = PR_HashTableAdd(_instructionHookTable,key,hook)?PR_TRUE:PR_FALSE; _unlock(); JSD_SetExecutionHook(_jsdc, (JSDScript*)pc.script.jsdscript, pc.offset, ssjsdExecutionHookProc, this); } return retval; } IJSExecutionHook_ptr DebugController::_removeInstructionHook(const IJSPC& pc) { JSD_ClearExecutionHook(_jsdc, (JSDScript*)pc.script.jsdscript, pc.offset ); IJSExecutionHook_ptr hook = NULL; if( _instructionHookTable ) { HookKey key(pc); _lock(); PRHashEntry** entry = PR_HashTableRawLookup(_instructionHookTable, key.hash_num, &key ); if( entry && *entry ) { IJSExecutionHook_ptr hook = (IJSExecutionHook_ptr)(*entry)->value; delete ((HookKey*)(*entry)->key); PR_HashTableRemove(_instructionHookTable, &key ); } _unlock(); } return hook; } PRBool DebugController::_initInstructionHookTable() { _instructionHookTable = PR_NewHashTable( 256, _hash_root, _hash_key_comparer, PR_CompareValues, NULL, NULL); return _instructionHookTable ? PR_TRUE : PR_FALSE; } void DebugController::_freeInstructionHookTable() { if( _instructionHookTable ) { PR_HashTableEnumerateEntries(_instructionHookTable, _hash_entry_zapper, NULL); PR_HashTableDestroy(_instructionHookTable); } _instructionHookTable = NULL; } void DebugController::_clearStepHandler(PRBool canReinstate) { if( canReinstate ) { if( _stepHandler ) { if( _oldStepHandler ) delete _oldStepHandler; _oldStepHandler = _stepHandler; _stepHandler = NULL; JSD_ClearInterruptHook(_jsdc); } } else { if( _stepHandler ) { delete _stepHandler; _stepHandler = NULL; JSD_ClearInterruptHook(_jsdc); } if( _oldStepHandler ) { delete _oldStepHandler; _oldStepHandler = NULL; } } } void DebugController::_reinstateStepHandler() { if( _oldStepHandler && ! _stepHandler ) { _stepHandler = _oldStepHandler; _oldStepHandler = NULL; JSD_SetInterruptHook(_jsdc, ssjsdExecutionHookProc, this ); } } /**************************/ HookKey::HookKey(const IJSPC& pc) { this->pc = pc; this->hash_num = ((long)pc.script.jsdscript) + (pc.offset*7); } /***************************************************************************/ /***************************************************************************/ PRMonitor* ThreadState::_monitor = NULL; PRHashTable* ThreadState::_thread2IndexTable = NULL; ThreadState::ThreadState(PRThread* currentThread) : _currentThread(currentThread), _controller(NULL), _hook(NULL), _pcTop(NULL), _isRunningEval(PR_FALSE), _isRunningHook(PR_FALSE), _actualCallToHookIsHappening(PR_FALSE), _leaveSuspended(PR_FALSE), _mainEventQueue(NULL), _hookCallerEventQueue(NULL), _hookCallerEventQueueCreateFinished(PR_FALSE), _suicideMessageReceived(PR_FALSE), _execResult() { // do nothing... } ThreadState::~ThreadState() { if( _mainEventQueue ) PR_DestroyEventQueue(_mainEventQueue); _removeIndexForThread(_currentThread); } PRHashNumber ThreadState::_hash_root(const void *key) { PRHashNumber num = (PRHashNumber) key; /* help lame MSVC1.5 on Win16 */ return num >> 2; } PRBool ThreadState::_prepThread2IndexTable() { if(_thread2IndexTable) return PR_TRUE; _lock(); // re check to avoid nasty race... if(! _thread2IndexTable) _thread2IndexTable = PR_NewHashTable( 256, _hash_root, PR_CompareValues, PR_CompareValues, NULL, NULL); _unlock(); PR_ASSERT(_thread2IndexTable); return _thread2IndexTable ? PR_TRUE : PR_FALSE; } PRBool ThreadState::_findIndexForThread(PRThread* thread, PRUintn* index) { PRUintn indexPlusOne; PR_ASSERT(thread); PR_ASSERT(index); if( ! _prepThread2IndexTable() ) return PR_FALSE; _lock(); indexPlusOne = (PRUintn) PR_HashTableLookup(_thread2IndexTable, thread); _unlock(); if(indexPlusOne) { *index = indexPlusOne - 1; return PR_TRUE; } return PR_FALSE; } PRBool ThreadState::_setIndexForThread(PRThread* thread, PRUintn index) { PRBool retval; PR_ASSERT(thread); if( ! _prepThread2IndexTable() ) return PR_FALSE; _lock(); retval = PR_HashTableAdd(_thread2IndexTable,thread,(void*)(index+1)) ? PR_TRUE : PR_FALSE; _unlock(); return retval; } PRBool ThreadState::_removeIndexForThread(PRThread* thread) { PRBool retval; PR_ASSERT(thread); if( ! _prepThread2IndexTable() ) return PR_FALSE; _lock(); retval = PR_HashTableRemove(_thread2IndexTable,thread)?PR_TRUE:PR_FALSE; _unlock(); return retval; } /**************************/ ThreadState* ThreadState::findThreadStateForCurrentThread(void) { return _findThreadState(PR_GetCurrentThread()); } ThreadState* ThreadState::getThreadStateForCurrentThread(void) { PRThread* currentThread = PR_GetCurrentThread(); ThreadState* ts = _findThreadState(currentThread); if(! ts) ts = _createThreadState(currentThread); return ts; } ThreadState* ThreadState::_findThreadState(PRThread* currentThread) { PRUintn index; PR_ASSERT(PR_GetCurrentThread() == currentThread); if( _findIndexForThread(currentThread, &index) ) return (ThreadState*) PR_GetThreadPrivate(index); return NULL; } ThreadState* ThreadState::_createThreadState(PRThread* currentThread) { PRUintn index; if(PR_FAILURE == PR_NewThreadPrivateIndex(&index,_callbackForPRThreadDeath)) return NULL; if( ! _setIndexForThread(currentThread, index) ) return NULL; ThreadState* ts = new ThreadState(currentThread); if( ! ts->_initEventQueues() ) return NULL; if( PR_FAILURE == PR_SetThreadPrivate(index, ts) ) return NULL; return ts; } /**************************/ PRBool ThreadState::_initEventQueues(void) { char name[64]; PR_ASSERT(NULL == _mainEventQueue); PR_ASSERT(NULL == _hookCallerEventQueue); PR_ASSERT(! _hookCallerEventQueueCreateFinished); PR_ASSERT(! _suicideMessageReceived); sprintf(name, "ThreadState::_mainEventQueue_%d", (int) this); _mainEventQueue = PR_CreateEventQueue(name, _currentThread); if( ! _mainEventQueue ) return PR_FALSE; // the _hookCallerEventQueue queue gets created on the 'other' thread. _lock(); if( PR_CreateThread(PR_USER_THREAD, _otherThreadProc, this, PR_PRIORITY_NORMAL, PR_GLOBAL_THREAD, PR_UNJOINABLE_THREAD, 0) ) { while( ! _hookCallerEventQueueCreateFinished ) { _wait(); } } _unlock(); return NULL == _hookCallerEventQueue ? PR_FALSE : PR_TRUE; } void ThreadState::_otherThreadProc(void* arg) { char name[64]; ThreadState* self = (ThreadState*)arg; sprintf(name, "ThreadState::_hookCallerEventQueue_%d", (int) self); _lock(); self->_hookCallerEventQueue = PR_CreateEventQueue(name, PR_GetCurrentThread()); self->_hookCallerEventQueueCreateFinished = PR_TRUE; _notify(); _unlock(); // run in the queue until we process the suicide message if(self->_hookCallerEventQueue) { // run the queue while( ! self->_suicideMessageReceived ) { PREvent* e = PR_WaitForEvent(self->_hookCallerEventQueue); if(e) { PR_HandleEvent(e); } } // delete our queue PR_DestroyEventQueue(self->_hookCallerEventQueue); self->_hookCallerEventQueue = NULL; // delete our self! delete self; } // return here ends this 'other' thread } void ThreadState::_callbackForPRThreadDeath(void *arg) { ThreadState* self = (ThreadState*)arg; if(self->_hookCallerEventQueue) { // post suicide message JSDSuicideEvent* e = new JSDSuicideEvent(self); e->post(self->_hookCallerEventQueue); } else { // otherwise, just delete it. delete self; } } /**************************/ void ThreadState::callHook(DebugController* controller, IJSExecutionHook* hook, sequence_of_IJSStackFrameInfo& stack, IJSPC* pcTop, JSDThreadState* jsdthreadstate) { PR_ASSERT(controller); PR_ASSERT(hook); PR_ASSERT(pcTop); PR_ASSERT(jsdthreadstate); PR_ASSERT(_mainEventQueue); PR_ASSERT(_hookCallerEventQueue); PR_ASSERT(! _isRunningHook); PR_ASSERT(! _actualCallToHookIsHappening); if( ! _mainEventQueue || ! _hookCallerEventQueue ) return; // setup IJSThreadState _jsts.stack = stack; _jsts.continueState = (CORBA::Long) JSD_HOOK_RETURN_CONTINUE; _jsts.returnValue = (const char*) NULL; // currently ignored _jsts.status = 0; // currently ignored _jsts.jsdthreadstate = (CORBA::Long) jsdthreadstate; _jsts.id = (CORBA::Long) this; _controller = controller; _hook = hook; _pcTop = pcTop; _leaveSuspended = PR_FALSE; _actualCallToHookIsHappening = PR_TRUE; _isRunningHook = PR_TRUE; // post hook caller event JSDCallHookEvent* e = new JSDCallHookEvent(this); e->post(_hookCallerEventQueue); // proccess events until a JSDResumeEvent clears _isRunningHook while( _isRunningHook ) { PREvent* e = PR_WaitForEvent(_mainEventQueue); if(e) { PR_HandleEvent(e); } } } PRBool ThreadState::isRunningHook(void) { return _isRunningHook; } PRBool ThreadState::isWaitingForResume(void) { if(_isRunningHook && _leaveSuspended && ! _actualCallToHookIsHappening) return PR_TRUE; return PR_FALSE; } // // the critical flags we _lock() to protect are: // _leaveSuspended // _actualCallToHookIsHappening // void ThreadState::leaveThreadSuspended(void) { _lock(); _leaveSuspended = PR_TRUE; _unlock(); } void ThreadState::resumeThread() { _lock(); PR_ASSERT(_isRunningHook); _leaveSuspended = PR_FALSE; if(! _actualCallToHookIsHappening) { PR_ASSERT(_mainEventQueue); JSDResumeEvent* e = new JSDResumeEvent(this); e->post(_mainEventQueue); } _unlock(); } void ThreadState::eventSaysSuicide(void) { _suicideMessageReceived = PR_TRUE; } void ThreadState::eventSaysResume(void) { PR_ASSERT(_isRunningHook); _isRunningHook = PR_FALSE; } // called on 'other' thread... void ThreadState::eventSaysCallHook(void) { PR_ASSERT(_isRunningHook); PR_ASSERT(_hook); try { TRACE("+++HOOK _hook->aboutToExecute() call hook"); _hook->aboutToExecute(_jsts, *_pcTop); TRACE("---HOOK _hook->aboutToExecute() call hook"); } catch(...) { // eat it... _jsts.continueState = JSD_HOOK_RETURN_CONTINUE; } _lock(); _actualCallToHookIsHappening = PR_FALSE; if(! _leaveSuspended) { JSDResumeEvent* e = new JSDResumeEvent(this); e->post(_mainEventQueue); } _unlock(); } /***************************************************************************/ /***************************************************************************/ JSDResumeEvent::JSDResumeEvent(ThreadState* ts) : _ts(ts) { PR_ASSERT(_ts); PR_InitEvent(this, NULL, static_handle, static_destroy); } void JSDResumeEvent::post(PREventQueue* queue) { PR_ASSERT(queue); // PR_ENTER_EVENT_QUEUE_MONITOR(queue); PR_PostEvent(queue, this); // PR_EXIT_EVENT_QUEUE_MONITOR(queue); } void* JSDResumeEvent::static_handle(PREvent* e) { JSDResumeEvent* self = (JSDResumeEvent*) e; PR_ASSERT(self); PR_ASSERT(self->_ts); self->_ts->eventSaysResume(); return NULL; } void JSDResumeEvent::static_destroy(PREvent* e) { JSDResumeEvent* self = (JSDResumeEvent*) e; PR_ASSERT(self); delete self; } /***************************************************************************/ JSDSuicideEvent::JSDSuicideEvent(ThreadState* ts) : _ts(ts) { PR_ASSERT(_ts); PR_InitEvent(this, NULL, static_handle, static_destroy); } void JSDSuicideEvent::post(PREventQueue* queue) { PR_ASSERT(queue); // PR_ENTER_EVENT_QUEUE_MONITOR(queue); PR_PostEvent(queue, this); // PR_EXIT_EVENT_QUEUE_MONITOR(queue); } void* JSDSuicideEvent::static_handle(PREvent* e) { JSDSuicideEvent* self = (JSDSuicideEvent*) e; PR_ASSERT(self); PR_ASSERT(self->_ts); self->_ts->eventSaysSuicide(); return NULL; } void JSDSuicideEvent::static_destroy(PREvent* e) { JSDSuicideEvent* self = (JSDSuicideEvent*) e; PR_ASSERT(self); delete self; } /***************************************************************************/ JSDCallHookEvent::JSDCallHookEvent(ThreadState* ts) : _ts(ts) { PR_ASSERT(_ts); PR_InitEvent(this, NULL, static_handle, static_destroy); } void JSDCallHookEvent::post(PREventQueue* queue) { PR_ASSERT(queue); // PR_ENTER_EVENT_QUEUE_MONITOR(queue); PR_PostEvent(queue, this); // PR_EXIT_EVENT_QUEUE_MONITOR(queue); } void* JSDCallHookEvent::static_handle(PREvent* e) { JSDCallHookEvent* self = (JSDCallHookEvent*) e; PR_ASSERT(self); PR_ASSERT(self->_ts); self->_ts->eventSaysCallHook(); return NULL; } void JSDCallHookEvent::static_destroy(PREvent* e) { JSDCallHookEvent* self = (JSDCallHookEvent*) e; PR_ASSERT(self); delete self; } /***************************************************************************/ JSDExecuteEvent::JSDExecuteEvent(JSDContext* jsdc, ThreadState* ts, const IJSStackFrameInfo& frame, const char* text, const char* filename, int lineno) : _jsdc(jsdc), _ts(ts), _frame(frame), _text(strdup(text)), _filename(strdup(filename)), // set below _lineno(lineno) { PR_ASSERT(_jsdc); PR_ASSERT(_ts); PR_ASSERT(_text); PR_ASSERT(_filename); PR_InitEvent(this, NULL, static_handle, static_destroy); } JSDExecuteEvent::~JSDExecuteEvent() { if(_text) free(_text); if(_filename) free(_filename); } IExecResult* JSDExecuteEvent::postSynchronous(PREventQueue* queue) { PR_ASSERT(queue); IExecResult* result = (IExecResult*) PR_PostSynchronousEvent(queue, this); return result; } void* JSDExecuteEvent::static_handle(PREvent* e) { JSDExecuteEvent* self = (JSDExecuteEvent*) e; PR_ASSERT(self); PR_ASSERT(self->_ts); return self->_handle(); } void JSDExecuteEvent::static_destroy(PREvent* e) { JSDExecuteEvent* self = (JSDExecuteEvent*) e; PR_ASSERT(self); delete self; } void* JSDExecuteEvent::_handle() { const IJSThreadState& jsts = _ts->getJSThreadState(); JSDThreadState* jsdthreadstate = (JSDThreadState*)(jsts.jsdthreadstate); JSDStackFrameInfo* jsdframe = (JSDStackFrameInfo*) _frame.jsdframe; JSString* jsstr; jsval rval; JSBool success; char* str = NULL; IExecResult* result = NULL; // clear error result _ts->getExecResult() = (const IExecResult&) IExecResult(); _ts->setRunningEval(PR_TRUE); success = JSD_EvaluateScriptInStackFrame(_jsdc, jsdthreadstate, jsdframe, _text, strlen(_text), _filename, _lineno, &rval); _ts->setRunningEval(PR_FALSE); result = new IExecResult((const IExecResult&) _ts->getExecResult()); if( success && ! JSVAL_IS_NULL(rval) && ! JSVAL_IS_VOID(rval) ) { jsstr = JSD_ValToStringInStackFrame(_jsdc,jsdthreadstate,jsdframe,rval); if( jsstr ) { str = (char*) malloc((jsstr->length+1)*sizeof(char)); if(str) { memcpy( str, jsstr->bytes, jsstr->length*sizeof(char)); str[jsstr->length] = 0; } } } result->result = str; return result; } /***************************************************************************/ CallChain::CallChain(JSDContext* jsdc, JSDThreadState* jsdthreadstate) : _jsdc(jsdc), _chain(NULL), _count(0) { PR_ASSERT(_jsdc); PR_ASSERT(jsdthreadstate); _count = (int) JSD_GetCountOfStackFrames(jsdc, jsdthreadstate); PR_ASSERT(_count); _chain = (JSDScript**) new char[_count*sizeof(JSDScript*)]; PR_ASSERT(_chain); JSDStackFrameInfo* jsdframe = JSD_GetStackFrame(jsdc, jsdthreadstate); for(int i = _count-1; i >= 0; i--) { PR_ASSERT(jsdframe); _chain[i] = JSD_GetScriptForStackFrame(_jsdc, jsdthreadstate, jsdframe); PR_ASSERT(_chain[i]); jsdframe = JSD_GetCallingStackFrame(jsdc, jsdthreadstate, jsdframe); } } CallChain::~CallChain() { if( _chain ) delete (char*) _chain; } CallChain::CompareResult CallChain::compare(const CallChain& other) { if( this == &other ) return EQUAL; if( NULL == _chain || NULL == other._chain || 0 == _count || 0 == other._count ) return DISJOINT; int lesser_count = min( _count, other._count ); for( int i = 0; i < lesser_count; i++ ) { if( _chain[i] != other._chain[i] ) return DISJOINT; } if( _count > other._count ) return CALLER; if( _count < other._count ) return CALLEE; return EQUAL; } /***************************************************************************/ StepHandler::StepHandler(JSDContext* jsdc, ThreadState* state, PRBool buildCallChain) : _jsdc(jsdc), topScriptInitial(NULL), topPCInitial(0), topLineInitial(0), _callChain(NULL) { PR_ASSERT(_jsdc); PR_ASSERT(state); PR_ASSERT(state->isRunningHook()); JSDThreadState* jsdthreadstate = (JSDThreadState*) state->getJSThreadState().jsdthreadstate; PR_ASSERT(jsdthreadstate); JSDStackFrameInfo* jsdframe = _topFrame(jsdthreadstate); PR_ASSERT(jsdframe); topScriptInitial = _topScript(jsdthreadstate, jsdframe); PR_ASSERT(topScriptInitial); topPCInitial = _topPC(jsdthreadstate, jsdframe); topLineInitial = _topLine(topScriptInitial, topPCInitial); if( buildCallChain ) { _callChain = new CallChain( _jsdc, jsdthreadstate ); PR_ASSERT(_callChain); } } StepHandler::~StepHandler() { if( _callChain ) delete _callChain; } /***************************************************************************/ StepInto::StepInto(JSDContext* jsdc, ThreadState* state) : StepHandler(jsdc, state, PR_FALSE) { // do nothing } StepHandler::StepResult StepInto::step(JSDThreadState* jsdthreadstate) { JSDStackFrameInfo* topFrame = _topFrame(jsdthreadstate); JSDScript* topScript = _topScript(jsdthreadstate, topFrame); // different script, gotta stop if( topScriptInitial != topScript ) return STOP; // // NOTE: This little dance is necessary because line numbers for PCs // are not always acending. e,g, in: // // for(i=0;i topLineInitial ) return STOP; return CONTINUE_SEND_INTERRUPT; } /***************************************************************************/ StepOver::StepOver(JSDContext* jsdc, ThreadState* state) : StepHandler(jsdc, state, PR_TRUE) { // do nothing } StepHandler::StepResult StepOver::step(JSDThreadState* jsdthreadstate) { switch( _callChain->compare( CallChain(_jsdc, jsdthreadstate) ) ) { case CallChain::EQUAL: break; case CallChain::CALLER: return STOP; case CallChain::CALLEE: return CONTINUE_SEND_INTERRUPT; case CallChain::DISJOINT: return STOP; default: PR_ASSERT(0); // illegal value! break; } JSDStackFrameInfo* topFrame = _topFrame(jsdthreadstate); JSDScript* topScript = _topScript(jsdthreadstate, topFrame); prword_t topPC = _topPC(jsdthreadstate, topFrame); int topLine = _topLine(topScript, topPC); // definitely jumping back if( topPC < topPCInitial && topLine != topLineInitial ) return STOP; // definitely jumping forward if( topLine > topLineInitial ) return STOP; return CONTINUE_SEND_INTERRUPT; } /***************************************************************************/ StepOut::StepOut(JSDContext* jsdc, ThreadState* state) : StepHandler(jsdc, state, PR_TRUE) { // do nothing } StepHandler::StepResult StepOut::step(JSDThreadState* jsdthreadstate) { switch( _callChain->compare( CallChain(_jsdc, jsdthreadstate) ) ) { case CallChain::EQUAL: return CONTINUE_SEND_INTERRUPT; case CallChain::CALLER: return STOP; case CallChain::CALLEE: return CONTINUE_SEND_INTERRUPT; case CallChain::DISJOINT: return STOP; default: PR_ASSERT(0); // illegal value! return STOP; } } /***************************************************************************/ /***************************************************************************/ SourceTextProvider::SourceTextProvider(const char* name) : _sk_ISourceTextProvider(name), _jsdc(NULL) { // do nothing } SourceTextProvider::SourceTextProvider(const char* name, JSDContext* jsdc) : _sk_ISourceTextProvider(name), _jsdc(NULL) { init(jsdc); } void SourceTextProvider::init(JSDContext* jsdc) { _jsdc = jsdc; _forcePreLoad(); } SourceTextProvider::~SourceTextProvider() { } ISourceTextProvider::sequence_of_string * SourceTextProvider::getAllPages() { TRACE("SourceTextProvider::getAllPages() called"); JSDSourceText* jsdsrc; JSDSourceText* iter; ISourceTextProvider::sequence_of_string * seq = NULL; _lock(); // iterate once to get count; int count = 0; iter = NULL; while( NULL != JSD_IterateSources(_jsdc, &iter) ) count++; char** pp = (char**) malloc(count * sizeof(void*)); if(pp) { int i = 0; iter = NULL; while( NULL != (jsdsrc = JSD_IterateSources(_jsdc, &iter)) ) { const char* url; if( NULL == (url = JSD_GetSourceURL(_jsdc, jsdsrc)) ) continue; pp[i++] = strdup(url); } seq = new ISourceTextProvider::sequence_of_string(i, i, pp, 1); } _unlock(); return seq; } void SourceTextProvider::refreshAllPages() { TRACE("SourceTextProvider::refreshAllPages() called"); _lock(); // what we really want is a way to detect if existing pages // have changed - date, size, etc... -- and force reload of them // JSD_DestroyAllSources(_jsdc); _forcePreLoad(); _unlock(); } CORBA::Boolean SourceTextProvider::hasPage(const char * arg0) { TRACE("SourceTextProvider::hasPage() called"); return JSD_FindSourceForURL(_jsdc, arg0) ? 1 : 0; } CORBA::Boolean SourceTextProvider::loadPage(const char * arg0) { TRACE("SourceTextProvider::loadPage() called"); // not implemented... return 0; } void SourceTextProvider::refreshPage(const char * arg0) { TRACE("SourceTextProvider::refreshPage() called"); // not implemented... } char * SourceTextProvider::getPageText(const char * arg0) { TRACE("SourceTextProvider::getPageText() called"); JSDSourceText* jsdsrc; const char * const_text; int len; char* text = NULL; _lock(); jsdsrc = JSD_FindSourceForURL(_jsdc, arg0); if( jsdsrc ) { if( JSD_SOURCE_COMPLETED != JSD_GetSourceStatus(_jsdc, jsdsrc) ) jsdsrc = JSDLW_ForceLoadSource(_jsdc, jsdsrc); } if( jsdsrc ) { if( JSD_GetSourceText(_jsdc, jsdsrc, &const_text, &len) && const_text && len ) { text = (char*) malloc((len+1)*sizeof(char)); if(text) { memcpy(text,const_text,len*sizeof(char)); text[len] = 0; _deferedFree(text); } } } _unlock(); return text; } CORBA::Long SourceTextProvider::getPageStatus(const char * arg0) { TRACE("SourceTextProvider::getPageStatus() called"); JSDSourceText* jsdsrc; int status = JSD_SOURCE_INITED; _lock(); jsdsrc = JSD_FindSourceForURL(_jsdc, arg0); if(jsdsrc) status = JSD_GetSourceStatus(_jsdc, jsdsrc); _unlock(); return (CORBA::Long) status; } CORBA::Long SourceTextProvider::getPageAlterCount(const char * arg0) { TRACE("SourceTextProvider::getPageAlterCount() called"); JSDSourceText* jsdsrc; int altercount = JSD_SOURCE_INITED; _lock(); jsdsrc = JSD_FindSourceForURL(_jsdc, arg0); if(jsdsrc) altercount = JSD_GetSourceAlterCount(_jsdc, jsdsrc); _unlock(); return (CORBA::Long) altercount; } void SourceTextProvider::_forcePreLoad() { LWDBGApp* app; LWDBGApp* app_iter; _lock(); // iterate through all pages of all apps to force preload app_iter = NULL; while(NULL != (app = LWDBG_EnumerateApps(&app_iter))) { const char* pagename; void* page_iter = NULL; while(NULL != (pagename = LWDBG_EnumeratePageNames(app,&page_iter))) { JSDLW_PreLoadSource(_jsdc, app, pagename, JS_FALSE ); } } _unlock(); } /***************************************************************************/ char * TestInterface_impl::getFirstAppInList() { static char buf[1024]; LWDBGApp* iter = NULL; LWDBGApp* app; app = LWDBG_EnumerateApps(&iter); if( app ) return ((char*) LWDBG_GetURI(app)) + 1; return NULL; } void TestInterface_impl::getAppNames(StringReciever_ptr sr) { LWDBGApp* iter = NULL; LWDBGApp* app; while( NULL != (app = LWDBG_EnumerateApps(&iter)) ) { char* name = ((char*) LWDBG_GetURI(app)) + 1; try { sr->recieveString(name); } catch(const CORBA::Exception&) { // eat it and bail... return; } } } static void InitThing(Thing& t, const char* s, int i) { t.s = s; t.i = i; } TestInterface::sequence_of_Thing * TestInterface_impl::getThings() { Thing* p = new Thing[3]; InitThing(p[0],"zero",0); InitThing(p[1],"one" ,1); InitThing(p[2],"two" ,2); return new TestInterface::sequence_of_Thing(3, 3, p, 1); } void TestInterface_impl::callBounce(StringReciever_ptr arg0, CORBA::Long arg1) { try { arg0->bounce(arg1); } catch(const CORBA::Exception&) { // eat it and bail... } } /***************************************************************************/ /***************************************************************************/ static JSDContext* _jsdc = NULL; static TestInterface_impl _test; static SourceTextProvider _stp("JSSourceTextProvider"); static DebugController _controller("JSDebugController"); static char* _hostname; static const char* _WAIReturnType2String( WAIReturnType_t t ) { switch(t) { case WAISPISuccess: return "WAISPISuccess"; case WAISPIFailure: return "WAISPIFailure"; case WAISPIBadparam: return "WAISPIBadparam"; case WAISPINonameservice: return "WAISPINonameservice"; default: return "unknown error code"; } } // PR_STATIC_CALLBACK(void) // ssjsdTestThreadProc(void* arg) // { // CORBA::ORB_var orb; // CORBA::BOA_var boa; // // int zero = 0; // orb = CORBA::ORB_init(zero,NULL); // boa = orb->BOA_init(zero,NULL); // // boa->obj_is_ready(&_test); // WAIReturnType_t reg = registerObject(_hostname, "test", &_test ); // // log_error(LOG_VERBOSE, "ssjsdTestThreadProc", NULL, NULL, // "registration of _test %s", // _WAIReturnType2String(reg) ); // // boa->impl_is_ready(); // } // // PR_STATIC_CALLBACK(void) // ssjsdDebugControllerThreadProc(void* arg) // { // CORBA::ORB_var orb; // CORBA::BOA_var boa; // // int zero = 0; // orb = CORBA::ORB_init(zero,NULL); // boa = orb->BOA_init(zero,NULL); // // _controller.init(_jsdc); // boa->obj_is_ready(&_controller); // WAIReturnType_t reg = registerObject(_hostname, "JSDebugController", &_controller ); // // log_error(LOG_VERBOSE, "ssjsdDebugControllerThreadProc", NULL, NULL, // "registration of DebugController %s", // _WAIReturnType2String(reg) ); // // boa->impl_is_ready(); // } // // PR_STATIC_CALLBACK(void) // ssjsdSourceTextProviderThreadProc(void* arg) // { // CORBA::ORB_var orb; // CORBA::BOA_var boa; // // int zero = 0; // orb = CORBA::ORB_init(zero,NULL); // boa = orb->BOA_init(zero,NULL); // // _stp.init(_jsdc); // boa->obj_is_ready(&_stp); // WAIReturnType_t reg = registerObject(_hostname, "JSSourceTextProvider", &_stp ); // // log_error(LOG_VERBOSE, "ssjsdSourceTextProviderThreadProc", NULL, NULL, // "registration of SourceTextProvider %s", // _WAIReturnType2String(reg) ); // boa->impl_is_ready(); // } // // static PRThread* _TestThread; // static PRThread* _DebugControllerThread; // static PRThread* _SourceTextProviderThread; // // static void // _exportRemotedObjectsOnSeparateThreads() // { // /////////////////////////// // // _TestThread = // PR_CreateThread( PR_USER_THREAD, // ssjsdTestThreadProc, // NULL, // PR_PRIORITY_NORMAL, // PR_GLOBAL_THREAD, // PR_UNJOINABLE_THREAD, // 0 ); // // log_error(LOG_VERBOSE, "ssjsdebugInit", sn, rq, // "create test thread %s", // _TestThread == NULL ? "failed" : "succeeded" ); // // /////////////////////////// // // _DebugControllerThread = // PR_CreateThread( PR_USER_THREAD, // ssjsdDebugControllerThreadProc, // NULL, // PR_PRIORITY_NORMAL, // PR_GLOBAL_THREAD, // PR_UNJOINABLE_THREAD, // 0 ); // // log_error(LOG_VERBOSE, "ssjsdebugInit", sn, rq, // "create debug controller thread %s", // _DebugControllerThread == NULL ? "failed" : "succeeded" ); // // // /////////////////////////// // // _SourceTextProviderThread = // PR_CreateThread( PR_USER_THREAD, // ssjsdSourceTextProviderThreadProc, // NULL, // PR_PRIORITY_NORMAL, // PR_GLOBAL_THREAD, // PR_UNJOINABLE_THREAD, // 0 ); // // log_error(LOG_VERBOSE, "ssjsdebugInit", sn, rq, // "create text provider thread %s", // _SourceTextProviderThread == NULL ? "failed" : "succeeded" ); // } static PRBool _registerOb(const char *name, CORBA::Object_ptr obj) { WAIReturnType_t reg = registerObject(_hostname, name, obj); log_error(LOG_VERBOSE, "ssjsdebugInit", NULL, NULL, "registration of %s returned %s", name, _WAIReturnType2String(reg)); return WAISPISuccess == reg ? PR_TRUE : PR_FALSE; } static void _exportRemotedObjectsOnThisThread() { CORBA::ORB_var orb; CORBA::BOA_var boa; int zero = 0; orb = CORBA::ORB_init(zero,NULL); boa = orb->BOA_init(zero,NULL); /////////////////////////// // boa->obj_is_ready(&_test); // _registerOb("test", &_test, force ); /////////////////////////// _controller.init(_jsdc); boa->obj_is_ready(&_controller); _registerOb("JSDebugController", &_controller); /////////////////////////// _stp.init(_jsdc); boa->obj_is_ready(&_stp); _registerOb("JSSourceTextProvider", &_stp); } /* NSAPI initialization function */ PR_IMPLEMENT(int) ssjsdebugInit(pblock *pb, Session *sn, Request *rq) { JSD_SetUserCallbacks(LWDBG_GetTaskState(), NULL, NULL); _jsdc = JSD_DebuggerOn(); log_error(LOG_VERBOSE, "ssjsdebugInit", NULL, NULL, "starting with iiop, JSD_DebuggerOn() %s", _jsdc == NULL ? "failed" : "succeeded" ); if(_jsdc) { _hostname = http_uri2url("",""); log_error(LOG_VERBOSE, "ssjsdebugInit", NULL, NULL, "hostname = %s", _hostname ); _exportRemotedObjectsOnThisThread(); } return REQ_PROCEED; }