2  * Copyright (C) 2008, 2009 Apple Inc. All rights reserved. 
   4  * Redistribution and use in source and binary forms, with or without 
   5  * modification, are permitted provided that the following conditions 
   8  * 1.  Redistributions of source code must retain the above copyright 
   9  *     notice, this list of conditions and the following disclaimer. 
  10  * 2.  Redistributions in binary form must reproduce the above copyright 
  11  *     notice, this list of conditions and the following disclaimer in the 
  12  *     documentation and/or other materials provided with the distribution. 
  13  * 3.  Neither the name of Apple Computer, Inc. ("Apple") nor the names of 
  14  *     its contributors may be used to endorse or promote products derived 
  15  *     from this software without specific prior written permission. 
  17  * THIS SOFTWARE IS PROVIDED BY APPLE AND ITS CONTRIBUTORS "AS IS" AND ANY 
  18  * EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED 
  19  * WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE 
  20  * DISCLAIMED. IN NO EVENT SHALL APPLE OR ITS CONTRIBUTORS BE LIABLE FOR ANY 
  21  * DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES 
  22  * (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; 
  23  * LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND 
  24  * ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT 
  25  * (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF 
  26  * THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE. 
  30 #include "ExceptionHelpers.h" 
  32 #include "CodeBlock.h" 
  33 #include "CallFrame.h" 
  34 #include "JSGlobalObjectFunctions.h" 
  36 #include "JSNotAnObject.h" 
  37 #include "Interpreter.h" 
  42 class InterruptedExecutionError 
: public JSObject 
{ 
  44     InterruptedExecutionError(JSGlobalData
* globalData
) 
  45         : JSObject(globalData
->interruptedExecutionErrorStructure
) 
  49     virtual ComplType 
exceptionType() const { return Interrupted
; } 
  51     virtual UString 
toString(ExecState
*) const { return "JavaScript execution exceeded timeout."; } 
  54 JSValue 
createInterruptedExecutionException(JSGlobalData
* globalData
) 
  56     return new (globalData
) InterruptedExecutionError(globalData
); 
  59 class TerminatedExecutionError 
: public JSObject 
{ 
  61     TerminatedExecutionError(JSGlobalData
* globalData
) 
  62         : JSObject(globalData
->terminatedExecutionErrorStructure
) 
  66     virtual ComplType 
exceptionType() const { return Terminated
; } 
  68     virtual UString 
toString(ExecState
*) const { return "JavaScript execution terminated."; } 
  71 JSValue 
createTerminatedExecutionException(JSGlobalData
* globalData
) 
  73     return new (globalData
) TerminatedExecutionError(globalData
); 
  76 static JSValue 
createError(ExecState
* exec
, ErrorType e
, const char* msg
) 
  78     return Error::create(exec
, e
, msg
, -1, -1, UString()); 
  81 JSValue 
createStackOverflowError(ExecState
* exec
) 
  83     return createError(exec
, RangeError
, "Maximum call stack size exceeded."); 
  86 JSValue 
createTypeError(ExecState
* exec
, const char* message
) 
  88     return createError(exec
, TypeError
, message
); 
  91 JSValue 
createUndefinedVariableError(ExecState
* exec
, const Identifier
& ident
, unsigned bytecodeOffset
, CodeBlock
* codeBlock
) 
  96     int line 
= codeBlock
->expressionRangeForBytecodeOffset(exec
, bytecodeOffset
, divotPoint
, startOffset
, endOffset
); 
  97     JSObject
* exception 
= Error::create(exec
, ReferenceError
, makeString("Can't find variable: ", ident
.ustring()), line
, codeBlock
->ownerExecutable()->sourceID(), codeBlock
->ownerExecutable()->sourceURL()); 
  98     exception
->putWithAttributes(exec
, Identifier(exec
, expressionBeginOffsetPropertyName
), jsNumber(exec
, divotPoint 
- startOffset
), ReadOnly 
| DontDelete
); 
  99     exception
->putWithAttributes(exec
, Identifier(exec
, expressionCaretOffsetPropertyName
), jsNumber(exec
, divotPoint
), ReadOnly 
| DontDelete
); 
 100     exception
->putWithAttributes(exec
, Identifier(exec
, expressionEndOffsetPropertyName
), jsNumber(exec
, divotPoint 
+ endOffset
), ReadOnly 
| DontDelete
); 
 104 static UString 
createErrorMessage(ExecState
* exec
, CodeBlock
* codeBlock
, int, int expressionStart
, int expressionStop
, JSValue value
, UString error
) 
 106     if (!expressionStop 
|| expressionStart 
> codeBlock
->source()->length()) 
 107         return makeString(value
.toString(exec
), " is ", error
); 
 108     if (expressionStart 
< expressionStop
) 
 109         return makeString("Result of expression '", codeBlock
->source()->getRange(expressionStart
, expressionStop
), "' [", value
.toString(exec
), "] is ", error
, "."); 
 111     // No range information, so give a few characters of context 
 112     const UChar
* data 
= codeBlock
->source()->data(); 
 113     int dataLength 
= codeBlock
->source()->length(); 
 114     int start 
= expressionStart
; 
 115     int stop 
= expressionStart
; 
 116     // Get up to 20 characters of context to the left and right of the divot, clamping to the line. 
 117     // then strip whitespace. 
 118     while (start 
> 0 && (expressionStart 
- start 
< 20) && data
[start 
- 1] != '\n') 
 120     while (start 
< (expressionStart 
- 1) && isStrWhiteSpace(data
[start
])) 
 122     while (stop 
< dataLength 
&& (stop 
- expressionStart 
< 20) && data
[stop
] != '\n') 
 124     while (stop 
> expressionStart 
&& isStrWhiteSpace(data
[stop
])) 
 126     return makeString("Result of expression near '...", codeBlock
->source()->getRange(start
, stop
), "...' [", value
.toString(exec
), "] is ", error
, "."); 
 129 JSObject
* createInvalidParamError(ExecState
* exec
, const char* op
, JSValue value
, unsigned bytecodeOffset
, CodeBlock
* codeBlock
) 
 134     int line 
= codeBlock
->expressionRangeForBytecodeOffset(exec
, bytecodeOffset
, divotPoint
, startOffset
, endOffset
); 
 135     UString errorMessage 
= createErrorMessage(exec
, codeBlock
, line
, divotPoint
, divotPoint 
+ endOffset
, value
, makeString("not a valid argument for '", op
, "'")); 
 136     JSObject
* exception 
= Error::create(exec
, TypeError
, errorMessage
, line
, codeBlock
->ownerExecutable()->sourceID(), codeBlock
->ownerExecutable()->sourceURL()); 
 137     exception
->putWithAttributes(exec
, Identifier(exec
, expressionBeginOffsetPropertyName
), jsNumber(exec
, divotPoint 
- startOffset
), ReadOnly 
| DontDelete
); 
 138     exception
->putWithAttributes(exec
, Identifier(exec
, expressionCaretOffsetPropertyName
), jsNumber(exec
, divotPoint
), ReadOnly 
| DontDelete
); 
 139     exception
->putWithAttributes(exec
, Identifier(exec
, expressionEndOffsetPropertyName
), jsNumber(exec
, divotPoint 
+ endOffset
), ReadOnly 
| DontDelete
); 
 143 JSObject
* createNotAConstructorError(ExecState
* exec
, JSValue value
, unsigned bytecodeOffset
, CodeBlock
* codeBlock
) 
 148     int line 
= codeBlock
->expressionRangeForBytecodeOffset(exec
, bytecodeOffset
, divotPoint
, startOffset
, endOffset
); 
 150     // We're in a "new" expression, so we need to skip over the "new.." part 
 151     int startPoint 
= divotPoint 
- (startOffset 
? startOffset 
- 4 : 0); // -4 for "new " 
 152     const UChar
* data 
= codeBlock
->source()->data(); 
 153     while (startPoint 
< divotPoint 
&& isStrWhiteSpace(data
[startPoint
])) 
 156     UString errorMessage 
= createErrorMessage(exec
, codeBlock
, line
, startPoint
, divotPoint
, value
, "not a constructor"); 
 157     JSObject
* exception 
= Error::create(exec
, TypeError
, errorMessage
, line
, codeBlock
->ownerExecutable()->sourceID(), codeBlock
->ownerExecutable()->sourceURL()); 
 158     exception
->putWithAttributes(exec
, Identifier(exec
, expressionBeginOffsetPropertyName
), jsNumber(exec
, divotPoint 
- startOffset
), ReadOnly 
| DontDelete
); 
 159     exception
->putWithAttributes(exec
, Identifier(exec
, expressionCaretOffsetPropertyName
), jsNumber(exec
, divotPoint
), ReadOnly 
| DontDelete
); 
 160     exception
->putWithAttributes(exec
, Identifier(exec
, expressionEndOffsetPropertyName
), jsNumber(exec
, divotPoint 
+ endOffset
), ReadOnly 
| DontDelete
); 
 164 JSValue 
createNotAFunctionError(ExecState
* exec
, JSValue value
, unsigned bytecodeOffset
, CodeBlock
* codeBlock
) 
 169     int line 
= codeBlock
->expressionRangeForBytecodeOffset(exec
, bytecodeOffset
, divotPoint
, startOffset
, endOffset
); 
 170     UString errorMessage 
= createErrorMessage(exec
, codeBlock
, line
, divotPoint 
- startOffset
, divotPoint
, value
, "not a function"); 
 171     JSObject
* exception 
= Error::create(exec
, TypeError
, errorMessage
, line
, codeBlock
->ownerExecutable()->sourceID(), codeBlock
->ownerExecutable()->sourceURL());     
 172     exception
->putWithAttributes(exec
, Identifier(exec
, expressionBeginOffsetPropertyName
), jsNumber(exec
, divotPoint 
- startOffset
), ReadOnly 
| DontDelete
); 
 173     exception
->putWithAttributes(exec
, Identifier(exec
, expressionCaretOffsetPropertyName
), jsNumber(exec
, divotPoint
), ReadOnly 
| DontDelete
); 
 174     exception
->putWithAttributes(exec
, Identifier(exec
, expressionEndOffsetPropertyName
), jsNumber(exec
, divotPoint 
+ endOffset
), ReadOnly 
| DontDelete
); 
 178 JSNotAnObjectErrorStub
* createNotAnObjectErrorStub(ExecState
* exec
, bool isNull
) 
 180     return new (exec
) JSNotAnObjectErrorStub(exec
, isNull
); 
 183 JSObject
* createNotAnObjectError(ExecState
* exec
, JSNotAnObjectErrorStub
* error
, unsigned bytecodeOffset
, CodeBlock
* codeBlock
) 
 185     // Both op_construct and op_instanceof require a use of op_get_by_id to get 
 186     // the prototype property from an object. The exception messages for exceptions 
 187     // thrown by these instances op_get_by_id need to reflect this. 
 188     OpcodeID followingOpcodeID
; 
 189     if (codeBlock
->getByIdExceptionInfoForBytecodeOffset(exec
, bytecodeOffset
, followingOpcodeID
)) { 
 190         ASSERT(followingOpcodeID 
== op_construct 
|| followingOpcodeID 
== op_instanceof
); 
 191         if (followingOpcodeID 
== op_construct
) 
 192             return createNotAConstructorError(exec
, error
->isNull() ? jsNull() : jsUndefined(), bytecodeOffset
, codeBlock
); 
 193         return createInvalidParamError(exec
, "instanceof", error
->isNull() ? jsNull() : jsUndefined(), bytecodeOffset
, codeBlock
); 
 199     int line 
= codeBlock
->expressionRangeForBytecodeOffset(exec
, bytecodeOffset
, divotPoint
, startOffset
, endOffset
); 
 200     UString errorMessage 
= createErrorMessage(exec
, codeBlock
, line
, divotPoint 
- startOffset
, divotPoint
, error
->isNull() ? jsNull() : jsUndefined(), "not an object"); 
 201     JSObject
* exception 
= Error::create(exec
, TypeError
, errorMessage
, line
, codeBlock
->ownerExecutable()->sourceID(), codeBlock
->ownerExecutable()->sourceURL()); 
 202     exception
->putWithAttributes(exec
, Identifier(exec
, expressionBeginOffsetPropertyName
), jsNumber(exec
, divotPoint 
- startOffset
), ReadOnly 
| DontDelete
); 
 203     exception
->putWithAttributes(exec
, Identifier(exec
, expressionCaretOffsetPropertyName
), jsNumber(exec
, divotPoint
), ReadOnly 
| DontDelete
); 
 204     exception
->putWithAttributes(exec
, Identifier(exec
, expressionEndOffsetPropertyName
), jsNumber(exec
, divotPoint 
+ endOffset
), ReadOnly 
| DontDelete
); 
 208 JSValue 
throwOutOfMemoryError(ExecState
* exec
) 
 210     return throwError(exec
, GeneralError
, "Out of memory");