1 // -*- c-basic-offset: 2 -*-
3 * Copyright (C) 1999-2002 Harri Porten (porten@kde.org)
4 * Copyright (C) 2001 Peter Kelly (pmk@post.com)
5 * Copyright (C) 2003, 2004, 2005, 2006, 2007 Apple Inc. All rights reserved.
6 * Copyright (C) 2007 Cameron Zwarich (cwzwarich@uwaterloo.ca)
7 * Copyright (C) 2007 Maks Orlovich
9 * This library is free software; you can redistribute it and/or
10 * modify it under the terms of the GNU Library General Public
11 * License as published by the Free Software Foundation; either
12 * version 2 of the License, or (at your option) any later version.
14 * This library is distributed in the hope that it will be useful,
15 * but WITHOUT ANY WARRANTY; without even the implied warranty of
16 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
17 * Library General Public License for more details.
19 * You should have received a copy of the GNU Library General Public License
20 * along with this library; see the file COPYING.LIB. If not, write to
21 * the Free Software Foundation, Inc., 51 Franklin Street, Fifth Floor,
22 * Boston, MA 02110-1301, USA.
29 #include "Activation.h"
30 #include "ExecState.h"
31 #include "JSGlobalObject.h"
33 #include "PropertyNameArray.h"
36 #include "function_object.h"
40 #include "operations.h"
41 #include "scope_chain_mark.h"
46 #include <wtf/ASCIICType.h>
47 #include <wtf/Assertions.h>
48 #include <wtf/MathExtras.h>
49 #include <wtf/unicode/UTF8.h>
52 using namespace Unicode
;
56 // ----------------------------- FunctionImp ----------------------------------
58 const ClassInfo
FunctionImp::info
= { "Function", &InternalFunctionImp::info
, 0 };
60 FunctionImp::FunctionImp(ExecState
* exec
, const Identifier
& name
, FunctionBodyNode
* b
, const ScopeChain
& sc
)
61 : InternalFunctionImp(exec
->lexicalGlobalObject()->functionPrototype(), name
)
67 void FunctionImp::mark()
69 InternalFunctionImp::mark();
73 JSValue
* FunctionImp::callAsFunction(ExecState
* exec
, JSObject
* thisObj
, const List
& args
)
75 FunctionExecState
newExec(exec
->dynamicGlobalObject(), thisObj
, body
.get(), exec
, this, args
);
76 JSValue
* result
= body
->execute(&newExec
);
77 if (newExec
.completionType() == Throw
) {
78 exec
->setException(result
);
81 if (newExec
.completionType() == ReturnValue
)
86 JSValue
* FunctionImp::argumentsGetter(ExecState
* exec
, JSObject
*, const Identifier
& propertyName
, const PropertySlot
& slot
)
88 FunctionImp
* thisObj
= static_cast<FunctionImp
*>(slot
.slotBase());
90 for (ExecState
* e
= exec
; e
; e
= e
->callingExecState())
91 if (e
->function() == thisObj
) {
92 e
->dynamicGlobalObject()->tearOffActivation(e
, e
!= exec
);
93 return e
->activationObject()->get(exec
, propertyName
);
99 JSValue
* FunctionImp::callerGetter(ExecState
* exec
, JSObject
*, const Identifier
&, const PropertySlot
& slot
)
101 FunctionImp
* thisObj
= static_cast<FunctionImp
*>(slot
.slotBase());
104 if (e
->function() == thisObj
)
106 e
= e
->callingExecState();
112 ExecState
* callingExecState
= e
->callingExecState();
113 if (!callingExecState
)
116 FunctionImp
* callingFunction
= callingExecState
->function();
117 if (!callingFunction
)
120 return callingFunction
;
123 JSValue
* FunctionImp::lengthGetter(ExecState
*, JSObject
*, const Identifier
&, const PropertySlot
& slot
)
125 FunctionImp
* thisObj
= static_cast<FunctionImp
*>(slot
.slotBase());
126 return jsNumber(thisObj
->body
->parameters().size());
129 bool FunctionImp::getOwnPropertySlot(ExecState
* exec
, const Identifier
& propertyName
, PropertySlot
& slot
)
131 // Find the arguments from the closest context.
132 if (propertyName
== exec
->propertyNames().arguments
) {
133 slot
.setCustom(this, argumentsGetter
);
137 // Compute length of parameters.
138 if (propertyName
== exec
->propertyNames().length
) {
139 slot
.setCustom(this, lengthGetter
);
143 if (propertyName
== exec
->propertyNames().caller
) {
144 slot
.setCustom(this, callerGetter
);
148 return InternalFunctionImp::getOwnPropertySlot(exec
, propertyName
, slot
);
151 void FunctionImp::put(ExecState
* exec
, const Identifier
& propertyName
, JSValue
* value
, int attr
)
153 if (propertyName
== exec
->propertyNames().arguments
|| propertyName
== exec
->propertyNames().length
)
155 InternalFunctionImp::put(exec
, propertyName
, value
, attr
);
158 bool FunctionImp::deleteProperty(ExecState
* exec
, const Identifier
& propertyName
)
160 if (propertyName
== exec
->propertyNames().arguments
|| propertyName
== exec
->propertyNames().length
)
162 return InternalFunctionImp::deleteProperty(exec
, propertyName
);
165 /* Returns the parameter name corresponding to the given index. eg:
166 * function f1(x, y, z): getParameterName(0) --> x
168 * If a name appears more than once, only the last index at which
169 * it appears associates with it. eg:
170 * function f2(x, x): getParameterName(0) --> null
172 Identifier
FunctionImp::getParameterName(int index
)
174 Vector
<Identifier
>& parameters
= body
->parameters();
176 if (static_cast<size_t>(index
) >= body
->parameters().size())
177 return CommonIdentifiers::shared()->nullIdentifier
;
179 Identifier name
= parameters
[index
];
181 // Are there any subsequent parameters with the same name?
182 size_t size
= parameters
.size();
183 for (size_t i
= index
+ 1; i
< size
; ++i
)
184 if (parameters
[i
] == name
)
185 return CommonIdentifiers::shared()->nullIdentifier
;
190 // ECMA 13.2.2 [[Construct]]
191 JSObject
* FunctionImp::construct(ExecState
* exec
, const List
& args
)
194 JSValue
* p
= get(exec
, exec
->propertyNames().prototype
);
196 proto
= static_cast<JSObject
*>(p
);
198 proto
= exec
->lexicalGlobalObject()->objectPrototype();
200 JSObject
* obj(new JSObject(proto
));
202 JSValue
* res
= call(exec
,obj
,args
);
205 return static_cast<JSObject
*>(res
);
210 // ------------------------------ IndexToNameMap ---------------------------------
212 // We map indexes in the arguments array to their corresponding argument names.
213 // Example: function f(x, y, z): arguments[0] = x, so we map 0 to Identifier("x").
215 // Once we have an argument name, we can get and set the argument's value in the
216 // activation object.
218 // We use Identifier::null to indicate that a given argument's value
219 // isn't stored in the activation object.
221 IndexToNameMap::IndexToNameMap(FunctionImp
* func
, const List
& args
)
223 _map
= new Identifier
[args
.size()];
224 this->size
= args
.size();
227 List::const_iterator end
= args
.end();
228 for (List::const_iterator it
= args
.begin(); it
!= end
; ++i
, ++it
)
229 _map
[i
] = func
->getParameterName(i
); // null if there is no corresponding parameter
232 IndexToNameMap::~IndexToNameMap()
237 bool IndexToNameMap::isMapped(const Identifier
& index
) const
240 unsigned indexAsNumber
= index
.toStrictUInt32(&indexIsNumber
);
245 if (indexAsNumber
>= size
)
248 if (_map
[indexAsNumber
].isNull())
254 void IndexToNameMap::unMap(const Identifier
& index
)
257 unsigned indexAsNumber
= index
.toStrictUInt32(&indexIsNumber
);
259 ASSERT(indexIsNumber
&& indexAsNumber
< size
);
261 _map
[indexAsNumber
] = CommonIdentifiers::shared()->nullIdentifier
;
264 Identifier
& IndexToNameMap::operator[](const Identifier
& index
)
267 unsigned indexAsNumber
= index
.toStrictUInt32(&indexIsNumber
);
269 ASSERT(indexIsNumber
&& indexAsNumber
< size
);
271 return _map
[indexAsNumber
];
274 // ------------------------------ Arguments ---------------------------------
276 const ClassInfo
Arguments::info
= { "Arguments", 0, 0 };
279 Arguments::Arguments(ExecState
* exec
, FunctionImp
* func
, const List
& args
, ActivationImp
* act
)
280 : JSObject(exec
->lexicalGlobalObject()->objectPrototype())
281 , _activationObject(act
)
282 , indexToNameMap(func
, args
)
284 putDirect(exec
->propertyNames().callee
, func
, DontEnum
);
285 putDirect(exec
->propertyNames().length
, args
.size(), DontEnum
);
288 List::const_iterator end
= args
.end();
289 for (List::const_iterator it
= args
.begin(); it
!= end
; ++it
, ++i
) {
290 Identifier name
= Identifier::from(i
);
291 if (!indexToNameMap
.isMapped(name
))
292 putDirect(name
, *it
, DontEnum
);
296 void Arguments::mark()
299 if (_activationObject
&& !_activationObject
->marked())
300 _activationObject
->mark();
303 JSValue
* Arguments::mappedIndexGetter(ExecState
* exec
, JSObject
*, const Identifier
& propertyName
, const PropertySlot
& slot
)
305 Arguments
* thisObj
= static_cast<Arguments
*>(slot
.slotBase());
306 return thisObj
->_activationObject
->get(exec
, thisObj
->indexToNameMap
[propertyName
]);
309 bool Arguments::getOwnPropertySlot(ExecState
* exec
, const Identifier
& propertyName
, PropertySlot
& slot
)
311 if (indexToNameMap
.isMapped(propertyName
)) {
312 slot
.setCustom(this, mappedIndexGetter
);
316 return JSObject::getOwnPropertySlot(exec
, propertyName
, slot
);
319 void Arguments::put(ExecState
* exec
, const Identifier
& propertyName
, JSValue
* value
, int attr
)
321 if (indexToNameMap
.isMapped(propertyName
)) {
322 _activationObject
->put(exec
, indexToNameMap
[propertyName
], value
, attr
);
324 JSObject::put(exec
, propertyName
, value
, attr
);
328 bool Arguments::deleteProperty(ExecState
* exec
, const Identifier
& propertyName
)
330 if (indexToNameMap
.isMapped(propertyName
)) {
331 indexToNameMap
.unMap(propertyName
);
334 return JSObject::deleteProperty(exec
, propertyName
);
338 // ------------------------------ ActivationImp --------------------------------
340 const ClassInfo
ActivationImp::info
= { "Activation", 0, 0 };
342 ActivationImp::ActivationImp(const ActivationData
& oldData
, bool leaveRelic
)
344 JSVariableObject::d
= new ActivationData(oldData
);
345 d()->leftRelic
= leaveRelic
;
348 ActivationImp::~ActivationImp()
354 void ActivationImp::init(ExecState
* exec
)
356 d()->symbolTable
= &exec
->function()->body
->symbolTable();
358 d()->function
= exec
->function();
359 d()->argumentsObject
= 0;
362 JSValue
* ActivationImp::argumentsGetter(ExecState
* exec
, JSObject
*, const Identifier
&, const PropertySlot
& slot
)
364 ActivationImp
* thisObj
= static_cast<ActivationImp
*>(slot
.slotBase());
366 if (!thisObj
->d()->argumentsObject
)
367 thisObj
->createArgumentsObject(exec
);
369 return thisObj
->d()->argumentsObject
;
372 PropertySlot::GetValueFunc
ActivationImp::getArgumentsGetter()
374 return ActivationImp::argumentsGetter
;
377 bool ActivationImp::getOwnPropertySlot(ExecState
* exec
, const Identifier
& propertyName
, PropertySlot
& slot
)
379 if (symbolTableGet(propertyName
, slot
))
382 if (JSValue
** location
= getDirectLocation(propertyName
)) {
383 slot
.setValueSlot(this, location
);
387 // Only return the built-in arguments object if it wasn't overridden above.
388 if (propertyName
== exec
->propertyNames().arguments
) {
389 for (ExecState
* e
= exec
; e
; e
= e
->callingExecState())
390 if (e
->function() == d()->function
) {
391 e
->dynamicGlobalObject()->tearOffActivation(e
, e
!= exec
);
392 ActivationImp
* newActivation
= e
->activationObject();
393 slot
.setCustom(newActivation
, newActivation
->getArgumentsGetter());
397 slot
.setCustom(this, getArgumentsGetter());
401 // We don't call through to JSObject because there's no way to give an
402 // activation object getter properties or a prototype.
403 ASSERT(!_prop
.hasGetterSetterProperties());
404 ASSERT(prototype() == jsNull());
408 bool ActivationImp::deleteProperty(ExecState
* exec
, const Identifier
& propertyName
)
410 if (propertyName
== exec
->propertyNames().arguments
)
413 return JSVariableObject::deleteProperty(exec
, propertyName
);
416 void ActivationImp::put(ExecState
*, const Identifier
& propertyName
, JSValue
* value
, int attr
)
418 // If any bits other than DontDelete are set, then we bypass the read-only check.
419 bool checkReadOnly
= !(attr
& ~DontDelete
);
420 if (symbolTablePut(propertyName
, value
, checkReadOnly
))
423 // We don't call through to JSObject because __proto__ and getter/setter
424 // properties are non-standard extensions that other implementations do not
425 // expose in the activation object.
426 ASSERT(!_prop
.hasGetterSetterProperties());
427 _prop
.put(propertyName
, value
, attr
, checkReadOnly
);
430 void ActivationImp::markChildren()
432 LocalStorage
& localStorage
= d()->localStorage
;
433 size_t size
= localStorage
.size();
435 for (size_t i
= 0; i
< size
; ++i
) {
436 JSValue
* value
= localStorage
[i
].value
;
438 if (!value
->marked())
442 if (!d()->function
->marked())
443 d()->function
->mark();
445 if (d()->argumentsObject
&& !d()->argumentsObject
->marked())
446 d()->argumentsObject
->mark();
449 void ActivationImp::mark()
455 void ActivationImp::createArgumentsObject(ExecState
* exec
)
457 // Since "arguments" is only accessible while a function is being called,
458 // we can retrieve our argument list from the ExecState for our function
459 // call instead of storing the list ourselves.
460 d()->argumentsObject
= new Arguments(exec
, d()->exec
->function(), *d()->exec
->arguments(), this);
463 ActivationImp::ActivationData::ActivationData(const ActivationData
& old
)
464 : JSVariableObjectData(old
)
466 , function(old
.function
)
467 , argumentsObject(old
.argumentsObject
)
472 // ------------------------------ Global Functions -----------------------------------
474 static JSValue
* encode(ExecState
* exec
, const List
& args
, const char* do_not_escape
)
476 UString r
= "", s
, str
= args
[0]->toString(exec
);
477 CString cstr
= str
.UTF8String(true);
479 return throwError(exec
, URIError
, "String contained an illegal UTF-16 sequence.");
480 const char* p
= cstr
.c_str();
481 for (size_t k
= 0; k
< cstr
.size(); k
++, p
++) {
483 if (c
&& strchr(do_not_escape
, c
)) {
487 snprintf(tmp
, sizeof(tmp
), "%%%02X", (unsigned char)c
);
494 static JSValue
* decode(ExecState
* exec
, const List
& args
, const char* do_not_unescape
, bool strict
)
496 UString s
= "", str
= args
[0]->toString(exec
);
497 int k
= 0, len
= str
.size();
498 const UChar
* d
= str
.data();
501 const UChar
* p
= d
+ k
;
505 if (k
<= len
- 3 && isASCIIHexDigit(p
[1].uc
) && isASCIIHexDigit(p
[2].uc
)) {
506 const char b0
= Lexer::convertHex(p
[1].uc
, p
[2].uc
);
507 const int sequenceLen
= UTF8SequenceLength(b0
);
508 if (sequenceLen
!= 0 && k
<= len
- sequenceLen
* 3) {
509 charLen
= sequenceLen
* 3;
512 for (int i
= 1; i
< sequenceLen
; ++i
) {
513 const UChar
* q
= p
+ i
* 3;
514 if (q
[0] == '%' && isASCIIHexDigit(q
[1].uc
) && isASCIIHexDigit(q
[2].uc
))
515 sequence
[i
] = Lexer::convertHex(q
[1].uc
, q
[2].uc
);
522 sequence
[sequenceLen
] = 0;
523 const int character
= decodeUTF8Sequence(sequence
);
524 if (character
< 0 || character
>= 0x110000) {
526 } else if (character
>= 0x10000) {
527 // Convert to surrogate pair.
528 s
.append(static_cast<unsigned short>(0xD800 | ((character
- 0x10000) >> 10)));
529 u
= static_cast<unsigned short>(0xDC00 | ((character
- 0x10000) & 0x3FF));
531 u
= static_cast<unsigned short>(character
);
538 return throwError(exec
, URIError
);
539 // The only case where we don't use "strict" mode is the "unescape" function.
540 // For that, it's good to support the wonky "%u" syntax for compatibility with WinIE.
541 if (k
<= len
- 6 && p
[1] == 'u'
542 && isASCIIHexDigit(p
[2].uc
) && isASCIIHexDigit(p
[3].uc
)
543 && isASCIIHexDigit(p
[4].uc
) && isASCIIHexDigit(p
[5].uc
)) {
545 u
= Lexer::convertUnicode(p
[2].uc
, p
[3].uc
, p
[4].uc
, p
[5].uc
);
548 if (charLen
&& (u
.uc
== 0 || u
.uc
>= 128 || !strchr(do_not_unescape
, u
.low()))) {
559 static bool isStrWhiteSpace(unsigned short c
)
573 return isSeparatorSpace(c
);
577 static int parseDigit(unsigned short c
, int radix
)
581 if (c
>= '0' && c
<= '9') {
583 } else if (c
>= 'A' && c
<= 'Z') {
584 digit
= c
- 'A' + 10;
585 } else if (c
>= 'a' && c
<= 'z') {
586 digit
= c
- 'a' + 10;
594 double parseIntOverflow(const char* s
, int length
, int radix
)
597 double radixMultiplier
= 1.0;
599 for (const char* p
= s
+ length
- 1; p
>= s
; p
--) {
600 if (radixMultiplier
== Inf
) {
606 int digit
= parseDigit(*p
, radix
);
607 number
+= digit
* radixMultiplier
;
610 radixMultiplier
*= radix
;
616 static double parseInt(const UString
& s
, int radix
)
618 int length
= s
.size();
621 while (p
< length
&& isStrWhiteSpace(s
[p
].uc
)) {
629 } else if (s
[p
] == '-') {
635 if ((radix
== 0 || radix
== 16) && length
- p
>= 2 && s
[p
] == '0' && (s
[p
+ 1] == 'x' || s
[p
+ 1] == 'X')) {
638 } else if (radix
== 0) {
639 if (p
< length
&& s
[p
] == '0')
645 if (radix
< 2 || radix
> 36)
648 int firstDigitPosition
= p
;
649 bool sawDigit
= false;
652 int digit
= parseDigit(s
[p
].uc
, radix
);
661 if (number
>= mantissaOverflowLowerBound
) {
663 number
= kjs_strtod(s
.substr(firstDigitPosition
, p
- firstDigitPosition
).ascii(), 0);
664 else if (radix
== 2 || radix
== 4 || radix
== 8 || radix
== 16 || radix
== 32)
665 number
= parseIntOverflow(s
.substr(firstDigitPosition
, p
- firstDigitPosition
).ascii(), p
- firstDigitPosition
, radix
);
671 return sign
* number
;
674 static double parseFloat(const UString
& s
)
676 // Check for 0x prefix here, because toDouble allows it, but we must treat it as 0.
677 // Need to skip any whitespace and then one + or - sign.
678 int length
= s
.size();
680 while (p
< length
&& isStrWhiteSpace(s
[p
].uc
)) {
683 if (p
< length
&& (s
[p
] == '+' || s
[p
] == '-')) {
686 if (length
- p
>= 2 && s
[p
] == '0' && (s
[p
+ 1] == 'x' || s
[p
+ 1] == 'X')) {
690 return s
.toDouble( true /*tolerant*/, false /* NaN for empty string */ );
693 JSValue
* globalFuncEval(ExecState
* exec
, JSObject
* thisObj
, const List
& args
)
695 JSValue
* x
= args
[0];
699 UString s
= x
->toString(exec
);
704 SourceCode source
= makeSource(s
);
705 RefPtr
<EvalNode
> evalNode
= parser().parse
<EvalNode
>(source
, &errLine
, &errMsg
);
707 // debugger code removed
709 // No program node means a syntax occurred
711 return throwError(exec
, SyntaxError
, errMsg
, errLine
, source
.provider()->asID(), NULL
);
713 bool switchGlobal
= thisObj
&& thisObj
!= exec
->dynamicGlobalObject() && thisObj
->isGlobalObject();
715 // enter a new execution context
716 exec
->dynamicGlobalObject()->tearOffActivation(exec
);
717 JSGlobalObject
* globalObject
= switchGlobal
? static_cast<JSGlobalObject
*>(thisObj
) : exec
->dynamicGlobalObject();
718 EvalExecState
newExec(globalObject
, evalNode
.get(), exec
);
721 newExec
.pushScope(thisObj
);
722 newExec
.setVariableObject(static_cast<JSGlobalObject
*>(thisObj
));
724 JSValue
* value
= evalNode
->execute(&newExec
);
728 if (newExec
.completionType() == Throw
) {
729 exec
->setException(value
);
733 return value
? value
: jsUndefined();
736 JSValue
* globalFuncParseInt(ExecState
* exec
, JSObject
*, const List
& args
)
738 return jsNumber(parseInt(args
[0]->toString(exec
), args
[1]->toInt32(exec
)));
741 JSValue
* globalFuncParseFloat(ExecState
* exec
, JSObject
*, const List
& args
)
743 return jsNumber(parseFloat(args
[0]->toString(exec
)));
746 JSValue
* globalFuncIsNaN(ExecState
* exec
, JSObject
*, const List
& args
)
748 return jsBoolean(isnan(args
[0]->toNumber(exec
)));
751 JSValue
* globalFuncIsFinite(ExecState
* exec
, JSObject
*, const List
& args
)
753 double n
= args
[0]->toNumber(exec
);
754 return jsBoolean(!isnan(n
) && !isinf(n
));
757 JSValue
* globalFuncDecodeURI(ExecState
* exec
, JSObject
*, const List
& args
)
759 static const char do_not_unescape_when_decoding_URI
[] =
762 return decode(exec
, args
, do_not_unescape_when_decoding_URI
, true);
765 JSValue
* globalFuncDecodeURIComponent(ExecState
* exec
, JSObject
*, const List
& args
)
767 return decode(exec
, args
, "", true);
770 JSValue
* globalFuncEncodeURI(ExecState
* exec
, JSObject
*, const List
& args
)
772 static const char do_not_escape_when_encoding_URI
[] =
773 "ABCDEFGHIJKLMNOPQRSTUVWXYZ"
774 "abcdefghijklmnopqrstuvwxyz"
776 "!#$&'()*+,-./:;=?@_~";
778 return encode(exec
, args
, do_not_escape_when_encoding_URI
);
781 JSValue
* globalFuncEncodeURIComponent(ExecState
* exec
, JSObject
*, const List
& args
)
783 static const char do_not_escape_when_encoding_URI_component
[] =
784 "ABCDEFGHIJKLMNOPQRSTUVWXYZ"
785 "abcdefghijklmnopqrstuvwxyz"
789 return encode(exec
, args
, do_not_escape_when_encoding_URI_component
);
792 JSValue
* globalFuncEscape(ExecState
* exec
, JSObject
*, const List
& args
)
794 static const char do_not_escape
[] =
795 "ABCDEFGHIJKLMNOPQRSTUVWXYZ"
796 "abcdefghijklmnopqrstuvwxyz"
800 UString r
= "", s
, str
= args
[0]->toString(exec
);
801 const UChar
* c
= str
.data();
802 for (int k
= 0; k
< str
.size(); k
++, c
++) {
806 snprintf(tmp
, sizeof(tmp
), "%%u%04X", u
);
808 } else if (u
!= 0 && strchr(do_not_escape
, (char)u
))
812 snprintf(tmp
, sizeof(tmp
), "%%%02X", u
);
821 JSValue
* globalFuncUnescape(ExecState
* exec
, JSObject
*, const List
& args
)
823 UString s
= "", str
= args
[0]->toString(exec
);
824 int k
= 0, len
= str
.size();
826 const UChar
* c
= str
.data() + k
;
828 if (*c
== UChar('%') && k
<= len
- 6 && *(c
+ 1) == UChar('u')) {
829 if (Lexer::isHexDigit((c
+ 2)->uc
) && Lexer::isHexDigit((c
+ 3)->uc
) && Lexer::isHexDigit((c
+ 4)->uc
) && Lexer::isHexDigit((c
+ 5)->uc
)) {
830 u
= Lexer::convertUnicode((c
+ 2)->uc
, (c
+ 3)->uc
, (c
+ 4)->uc
, (c
+ 5)->uc
);
834 } else if (*c
== UChar('%') && k
<= len
- 3 && Lexer::isHexDigit((c
+ 1)->uc
) && Lexer::isHexDigit((c
+ 2)->uc
)) {
835 u
= UChar(Lexer::convertHex((c
+1)->uc
, (c
+2)->uc
));
847 JSValue
* globalFuncKJSPrint(ExecState
* exec
, JSObject
*, const List
& args
)
849 puts(args
[0]->toString(exec
).ascii());
850 return jsUndefined();
854 // ------------------------------ PrototypeFunction -------------------------------
856 PrototypeFunction::PrototypeFunction(ExecState
* exec
, int len
, const Identifier
& name
, JSMemberFunction function
)
857 : InternalFunctionImp(exec
->lexicalGlobalObject()->functionPrototype(), name
)
858 , m_function(function
)
860 ASSERT_ARG(function
, function
);
861 putDirect(exec
->propertyNames().length
, jsNumber(len
), DontDelete
| ReadOnly
| DontEnum
);
864 PrototypeFunction::PrototypeFunction(ExecState
* exec
, FunctionPrototype
* functionPrototype
, int len
, const Identifier
& name
, JSMemberFunction function
)
865 : InternalFunctionImp(functionPrototype
, name
)
866 , m_function(function
)
868 ASSERT_ARG(function
, function
);
869 putDirect(exec
->propertyNames().length
, jsNumber(len
), DontDelete
| ReadOnly
| DontEnum
);
872 JSValue
* PrototypeFunction::callAsFunction(ExecState
* exec
, JSObject
* thisObj
, const List
& args
)
874 return m_function(exec
, thisObj
, args
);