| /* |
| * Copyright (C) 2005 Apple Computer, Inc. All rights reserved. |
| * |
| * Redistribution and use in source and binary forms, with or without |
| * modification, are permitted provided that the following conditions |
| * are met: |
| * |
| * 1. Redistributions of source code must retain the above copyright |
| * notice, this list of conditions and the following disclaimer. |
| * 2. Redistributions in binary form must reproduce the above copyright |
| * notice, this list of conditions and the following disclaimer in the |
| * documentation and/or other materials provided with the distribution. |
| * 3. Neither the name of Apple Computer, Inc. ("Apple") nor the names of |
| * its contributors may be used to endorse or promote products derived |
| * from this software without specific prior written permission. |
| * |
| * THIS SOFTWARE IS PROVIDED BY APPLE AND ITS CONTRIBUTORS "AS IS" AND ANY |
| * EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED |
| * WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE |
| * DISCLAIMED. IN NO EVENT SHALL APPLE OR ITS CONTRIBUTORS BE LIABLE FOR ANY |
| * DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES |
| * (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; |
| * LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND |
| * ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT |
| * (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF |
| * THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE. |
| */ |
| |
| #import "WebDataSource.h" |
| #import "WebDataSourceInternal.h" |
| #import "WebFrameInternal.h" |
| #import "WebScriptDebugDelegate.h" |
| #import "WebScriptDebugger.h" |
| #import "WebViewInternal.h" |
| #import <WebCore/Frame.h> |
| #import <WebCore/ScriptController.h> |
| #import <WebCore/WebScriptObjectPrivate.h> |
| #import <WebCore/runtime_root.h> |
| #import <debugger/Debugger.h> |
| #import <debugger/DebuggerActivation.h> |
| #import <debugger/DebuggerCallFrame.h> |
| #import <interpreter/CallFrame.h> |
| #import <runtime/Completion.h> |
| #import <runtime/JSFunction.h> |
| #import <runtime/JSGlobalObject.h> |
| #import <runtime/JSLock.h> |
| |
| using namespace JSC; |
| using namespace WebCore; |
| |
| // FIXME: these error strings should be public for future use by WebScriptObject and in WebScriptObject.h |
| NSString * const WebScriptErrorDomain = @"WebScriptErrorDomain"; |
| NSString * const WebScriptErrorDescriptionKey = @"WebScriptErrorDescription"; |
| NSString * const WebScriptErrorLineNumberKey = @"WebScriptErrorLineNumber"; |
| |
| @interface WebScriptCallFrame (WebScriptDebugDelegateInternal) |
| |
| - (id)_convertValueToObjcValue:(JSValue)value; |
| |
| @end |
| |
| @interface WebScriptCallFramePrivate : NSObject { |
| @public |
| WebScriptObject *globalObject; // the global object's proxy (not retained) |
| WebScriptCallFrame *caller; // previous stack frame |
| DebuggerCallFrame* debuggerCallFrame; |
| WebScriptDebugger* debugger; |
| } |
| @end |
| |
| @implementation WebScriptCallFramePrivate |
| - (void)dealloc |
| { |
| [caller release]; |
| delete debuggerCallFrame; |
| [super dealloc]; |
| } |
| @end |
| |
| // WebScriptCallFrame |
| // |
| // One of these is created to represent each stack frame. Additionally, there is a "global" |
| // frame to represent the outermost scope. This global frame is always the last frame in |
| // the chain of callers. |
| // |
| // The delegate can assign a "wrapper" to each frame object so it can relay calls through its |
| // own exported interface. This class is private to WebCore (and the delegate). |
| |
| @implementation WebScriptCallFrame (WebScriptDebugDelegateInternal) |
| |
| - (WebScriptCallFrame *)_initWithGlobalObject:(WebScriptObject *)globalObj debugger:(WebScriptDebugger *)debugger caller:(WebScriptCallFrame *)caller debuggerCallFrame:(const DebuggerCallFrame&)debuggerCallFrame |
| { |
| if ((self = [super init])) { |
| _private = [[WebScriptCallFramePrivate alloc] init]; |
| _private->globalObject = globalObj; |
| _private->caller = [caller retain]; |
| _private->debugger = debugger; |
| } |
| return self; |
| } |
| |
| - (void)_setDebuggerCallFrame:(const DebuggerCallFrame&)debuggerCallFrame |
| { |
| if (!_private->debuggerCallFrame) |
| _private->debuggerCallFrame = new DebuggerCallFrame(debuggerCallFrame); |
| else |
| *_private->debuggerCallFrame = debuggerCallFrame; |
| } |
| |
| - (void)_clearDebuggerCallFrame |
| { |
| delete _private->debuggerCallFrame; |
| _private->debuggerCallFrame = 0; |
| } |
| |
| - (id)_convertValueToObjcValue:(JSValue)value |
| { |
| if (!value) |
| return nil; |
| |
| WebScriptObject *globalObject = _private->globalObject; |
| if (value == [globalObject _imp]) |
| return globalObject; |
| |
| Bindings::RootObject* root1 = [globalObject _originRootObject]; |
| if (!root1) |
| return nil; |
| |
| Bindings::RootObject* root2 = [globalObject _rootObject]; |
| if (!root2) |
| return nil; |
| |
| return [WebScriptObject _convertValueToObjcValue:value originRootObject:root1 rootObject:root2]; |
| } |
| |
| @end |
| |
| |
| |
| @implementation WebScriptCallFrame |
| |
| - (void) dealloc |
| { |
| [_userInfo release]; |
| [_private release]; |
| [super dealloc]; |
| } |
| |
| - (void)setUserInfo:(id)userInfo |
| { |
| if (userInfo != _userInfo) { |
| [_userInfo release]; |
| _userInfo = [userInfo retain]; |
| } |
| } |
| |
| - (id)userInfo |
| { |
| return _userInfo; |
| } |
| |
| - (WebScriptCallFrame *)caller |
| { |
| return _private->caller; |
| } |
| |
| // Returns an array of scope objects (most local first). |
| // The properties of each scope object are the variables for that scope. |
| // Note that the last entry in the array will _always_ be the global object (windowScriptObject), |
| // whose properties are the global variables. |
| |
| - (NSArray *)scopeChain |
| { |
| if (!_private->debuggerCallFrame) |
| return [NSArray array]; |
| |
| JSLock lock(SilenceAssertionsOnly); |
| |
| ScopeChainNode* scopeChain = _private->debuggerCallFrame->scopeChain(); |
| if (!scopeChain->next) // global frame |
| return [NSArray arrayWithObject:_private->globalObject]; |
| |
| NSMutableArray *scopes = [[NSMutableArray alloc] init]; |
| |
| ScopeChainIterator end = scopeChain->end(); |
| for (ScopeChainIterator it = scopeChain->begin(); it != end; ++it) { |
| JSObject* object = it->get(); |
| if (object->isActivationObject()) |
| object = new (scopeChain->globalData) DebuggerActivation(*scopeChain->globalData, object); |
| [scopes addObject:[self _convertValueToObjcValue:object]]; |
| } |
| |
| NSArray *result = [NSArray arrayWithArray:scopes]; |
| [scopes release]; |
| return result; |
| } |
| |
| // Returns the name of the function for this frame, if available. |
| // Returns nil for anonymous functions and for the global frame. |
| |
| - (NSString *)functionName |
| { |
| if (!_private->debuggerCallFrame) |
| return nil; |
| |
| const UString* functionName = _private->debuggerCallFrame->functionName(); |
| return functionName ? toNSString(*functionName) : nil; |
| } |
| |
| // Returns the pending exception for this frame (nil if none). |
| |
| - (id)exception |
| { |
| if (!_private->debuggerCallFrame) |
| return nil; |
| |
| JSValue exception = _private->debuggerCallFrame->exception(); |
| return exception ? [self _convertValueToObjcValue:exception] : nil; |
| } |
| |
| // Evaluate some JavaScript code in the context of this frame. |
| // The code is evaluated as if by "eval", and the result is returned. |
| // If there is an (uncaught) exception, it is returned as though _it_ were the result. |
| // Calling this method on the global frame is not quite the same as calling the WebScriptObject |
| // method of the same name, due to the treatment of exceptions. |
| |
| - (id)evaluateWebScript:(NSString *)script |
| { |
| if (!_private->debuggerCallFrame) |
| return nil; |
| |
| JSLock lock(SilenceAssertionsOnly); |
| |
| // If this is the global call frame and there is no dynamic global object, |
| // Dashcode is attempting to execute JS in the evaluator using a stale |
| // WebScriptCallFrame. Instead, we need to set the dynamic global object |
| // and evaluate the JS in the global object's global call frame. |
| JSGlobalObject* globalObject = _private->debugger->globalObject(); |
| if (self == _private->debugger->globalCallFrame() && !globalObject->globalData().dynamicGlobalObject) { |
| JSGlobalObject* globalObject = _private->debugger->globalObject(); |
| |
| DynamicGlobalObjectScope globalObjectScope(globalObject->globalData(), globalObject); |
| |
| JSValue exception; |
| JSValue result = evaluateInGlobalCallFrame(stringToUString(script), exception, globalObject); |
| if (exception) |
| return [self _convertValueToObjcValue:exception]; |
| return result ? [self _convertValueToObjcValue:result] : nil; |
| } |
| |
| JSValue exception; |
| JSValue result = _private->debuggerCallFrame->evaluate(stringToUString(script), exception); |
| if (exception) |
| return [self _convertValueToObjcValue:exception]; |
| return result ? [self _convertValueToObjcValue:result] : nil; |
| } |
| |
| @end |