2 * Copyright (C) 2006 Apple Computer, 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
7 * 1. Redistributions of source code must retain the above copyright
8 * notice, this list of conditions and the following disclaimer.
9 * 2. Redistributions in binary form must reproduce the above copyright
10 * notice, this list of conditions and the following disclaimer in the
11 * documentation and/or other materials provided with the distribution.
13 * THIS SOFTWARE IS PROVIDED BY APPLE COMPUTER, INC. ``AS IS'' AND ANY
14 * EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
15 * IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR
16 * PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL APPLE COMPUTER, INC. OR
17 * CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL,
18 * EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO,
19 * PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR
20 * PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY
21 * OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
22 * (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE
23 * OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
26 #include "JavaScriptCore.h"
27 #include "JSBasePrivate.h"
28 #include "JSContextRefPrivate.h"
29 #include "JSObjectRefPrivate.h"
30 #include "JSScriptRefPrivate.h"
31 #include "JSStringRefPrivate.h"
33 #define ASSERT_DISABLED 0
34 #include <wtf/Assertions.h>
36 #if PLATFORM(MAC) || PLATFORM(IOS)
37 #include <mach/mach.h>
38 #include <mach/mach_time.h>
48 #include <wtf/MathExtras.h>
50 static double nan(const char*)
52 return std::numeric_limits
<double>::quiet_NaN();
60 #if JSC_OBJC_API_ENABLED
61 void testObjectiveCAPI(void);
64 extern void JSSynchronousGarbageCollectForDebugging(JSContextRef
);
66 static JSGlobalContextRef context
;
68 static void assertEqualsAsBoolean(JSValueRef value
, bool expectedValue
)
70 if (JSValueToBoolean(context
, value
) != expectedValue
) {
71 fprintf(stderr
, "assertEqualsAsBoolean failed: %p, %d\n", value
, expectedValue
);
76 static void assertEqualsAsNumber(JSValueRef value
, double expectedValue
)
78 double number
= JSValueToNumber(context
, value
, NULL
);
80 // FIXME <rdar://4668451> - On i386 the isnan(double) macro tries to map to the isnan(float) function,
81 // causing a build break with -Wshorten-64-to-32 enabled. The issue is known by the appropriate team.
82 // After that's resolved, we can remove these casts
83 if (number
!= expectedValue
&& !(isnan((float)number
) && isnan((float)expectedValue
))) {
84 fprintf(stderr
, "assertEqualsAsNumber failed: %p, %lf\n", value
, expectedValue
);
89 static void assertEqualsAsUTF8String(JSValueRef value
, const char* expectedValue
)
91 JSStringRef valueAsString
= JSValueToStringCopy(context
, value
, NULL
);
93 size_t jsSize
= JSStringGetMaximumUTF8CStringSize(valueAsString
);
94 char* jsBuffer
= (char*)malloc(jsSize
);
95 JSStringGetUTF8CString(valueAsString
, jsBuffer
, jsSize
);
98 for (i
= 0; jsBuffer
[i
]; i
++) {
99 if (jsBuffer
[i
] != expectedValue
[i
]) {
100 fprintf(stderr
, "assertEqualsAsUTF8String failed at character %d: %c(%d) != %c(%d)\n", i
, jsBuffer
[i
], jsBuffer
[i
], expectedValue
[i
], expectedValue
[i
]);
105 if (jsSize
< strlen(jsBuffer
) + 1) {
106 fprintf(stderr
, "assertEqualsAsUTF8String failed: jsSize was too small\n");
111 JSStringRelease(valueAsString
);
114 static void assertEqualsAsCharactersPtr(JSValueRef value
, const char* expectedValue
)
116 JSStringRef valueAsString
= JSValueToStringCopy(context
, value
, NULL
);
118 size_t jsLength
= JSStringGetLength(valueAsString
);
119 const JSChar
* jsBuffer
= JSStringGetCharactersPtr(valueAsString
);
121 CFStringRef expectedValueAsCFString
= CFStringCreateWithCString(kCFAllocatorDefault
,
123 kCFStringEncodingUTF8
);
124 CFIndex cfLength
= CFStringGetLength(expectedValueAsCFString
);
125 UniChar
* cfBuffer
= (UniChar
*)malloc(cfLength
* sizeof(UniChar
));
126 CFStringGetCharacters(expectedValueAsCFString
, CFRangeMake(0, cfLength
), cfBuffer
);
127 CFRelease(expectedValueAsCFString
);
129 if (memcmp(jsBuffer
, cfBuffer
, cfLength
* sizeof(UniChar
)) != 0) {
130 fprintf(stderr
, "assertEqualsAsCharactersPtr failed: jsBuffer != cfBuffer\n");
134 if (jsLength
!= (size_t)cfLength
) {
135 fprintf(stderr
, "assertEqualsAsCharactersPtr failed: jsLength(%ld) != cfLength(%ld)\n", jsLength
, cfLength
);
140 JSStringRelease(valueAsString
);
143 static bool timeZoneIsPST()
145 char timeZoneName
[70];
147 memset(>m
, 0, sizeof(gtm
));
148 strftime(timeZoneName
, sizeof(timeZoneName
), "%Z", >m
);
150 return 0 == strcmp("PST", timeZoneName
);
153 static JSValueRef jsGlobalValue
; // non-stack value for testing JSValueProtect()
155 /* MyObject pseudo-class */
157 static bool MyObject_hasProperty(JSContextRef context
, JSObjectRef object
, JSStringRef propertyName
)
159 UNUSED_PARAM(context
);
160 UNUSED_PARAM(object
);
162 if (JSStringIsEqualToUTF8CString(propertyName
, "alwaysOne")
163 || JSStringIsEqualToUTF8CString(propertyName
, "cantFind")
164 || JSStringIsEqualToUTF8CString(propertyName
, "throwOnGet")
165 || JSStringIsEqualToUTF8CString(propertyName
, "myPropertyName")
166 || JSStringIsEqualToUTF8CString(propertyName
, "hasPropertyLie")
167 || JSStringIsEqualToUTF8CString(propertyName
, "0")) {
174 static JSValueRef
MyObject_getProperty(JSContextRef context
, JSObjectRef object
, JSStringRef propertyName
, JSValueRef
* exception
)
176 UNUSED_PARAM(context
);
177 UNUSED_PARAM(object
);
179 if (JSStringIsEqualToUTF8CString(propertyName
, "alwaysOne")) {
180 return JSValueMakeNumber(context
, 1);
183 if (JSStringIsEqualToUTF8CString(propertyName
, "myPropertyName")) {
184 return JSValueMakeNumber(context
, 1);
187 if (JSStringIsEqualToUTF8CString(propertyName
, "cantFind")) {
188 return JSValueMakeUndefined(context
);
191 if (JSStringIsEqualToUTF8CString(propertyName
, "hasPropertyLie")) {
195 if (JSStringIsEqualToUTF8CString(propertyName
, "throwOnGet")) {
196 return JSEvaluateScript(context
, JSStringCreateWithUTF8CString("throw 'an exception'"), object
, JSStringCreateWithUTF8CString("test script"), 1, exception
);
199 if (JSStringIsEqualToUTF8CString(propertyName
, "0")) {
200 *exception
= JSValueMakeNumber(context
, 1);
201 return JSValueMakeNumber(context
, 1);
204 return JSValueMakeNull(context
);
207 static bool MyObject_setProperty(JSContextRef context
, JSObjectRef object
, JSStringRef propertyName
, JSValueRef value
, JSValueRef
* exception
)
209 UNUSED_PARAM(context
);
210 UNUSED_PARAM(object
);
212 UNUSED_PARAM(exception
);
214 if (JSStringIsEqualToUTF8CString(propertyName
, "cantSet"))
215 return true; // pretend we set the property in order to swallow it
217 if (JSStringIsEqualToUTF8CString(propertyName
, "throwOnSet")) {
218 JSEvaluateScript(context
, JSStringCreateWithUTF8CString("throw 'an exception'"), object
, JSStringCreateWithUTF8CString("test script"), 1, exception
);
224 static bool MyObject_deleteProperty(JSContextRef context
, JSObjectRef object
, JSStringRef propertyName
, JSValueRef
* exception
)
226 UNUSED_PARAM(context
);
227 UNUSED_PARAM(object
);
229 if (JSStringIsEqualToUTF8CString(propertyName
, "cantDelete"))
232 if (JSStringIsEqualToUTF8CString(propertyName
, "throwOnDelete")) {
233 JSEvaluateScript(context
, JSStringCreateWithUTF8CString("throw 'an exception'"), object
, JSStringCreateWithUTF8CString("test script"), 1, exception
);
240 static void MyObject_getPropertyNames(JSContextRef context
, JSObjectRef object
, JSPropertyNameAccumulatorRef propertyNames
)
242 UNUSED_PARAM(context
);
243 UNUSED_PARAM(object
);
245 JSStringRef propertyName
;
247 propertyName
= JSStringCreateWithUTF8CString("alwaysOne");
248 JSPropertyNameAccumulatorAddName(propertyNames
, propertyName
);
249 JSStringRelease(propertyName
);
251 propertyName
= JSStringCreateWithUTF8CString("myPropertyName");
252 JSPropertyNameAccumulatorAddName(propertyNames
, propertyName
);
253 JSStringRelease(propertyName
);
256 static JSValueRef
MyObject_callAsFunction(JSContextRef context
, JSObjectRef object
, JSObjectRef thisObject
, size_t argumentCount
, const JSValueRef arguments
[], JSValueRef
* exception
)
258 UNUSED_PARAM(context
);
259 UNUSED_PARAM(object
);
260 UNUSED_PARAM(thisObject
);
261 UNUSED_PARAM(exception
);
263 if (argumentCount
> 0 && JSValueIsString(context
, arguments
[0]) && JSStringIsEqualToUTF8CString(JSValueToStringCopy(context
, arguments
[0], 0), "throwOnCall")) {
264 JSEvaluateScript(context
, JSStringCreateWithUTF8CString("throw 'an exception'"), object
, JSStringCreateWithUTF8CString("test script"), 1, exception
);
265 return JSValueMakeUndefined(context
);
268 if (argumentCount
> 0 && JSValueIsStrictEqual(context
, arguments
[0], JSValueMakeNumber(context
, 0)))
269 return JSValueMakeNumber(context
, 1);
271 return JSValueMakeUndefined(context
);
274 static JSObjectRef
MyObject_callAsConstructor(JSContextRef context
, JSObjectRef object
, size_t argumentCount
, const JSValueRef arguments
[], JSValueRef
* exception
)
276 UNUSED_PARAM(context
);
277 UNUSED_PARAM(object
);
279 if (argumentCount
> 0 && JSValueIsString(context
, arguments
[0]) && JSStringIsEqualToUTF8CString(JSValueToStringCopy(context
, arguments
[0], 0), "throwOnConstruct")) {
280 JSEvaluateScript(context
, JSStringCreateWithUTF8CString("throw 'an exception'"), object
, JSStringCreateWithUTF8CString("test script"), 1, exception
);
284 if (argumentCount
> 0 && JSValueIsStrictEqual(context
, arguments
[0], JSValueMakeNumber(context
, 0)))
285 return JSValueToObject(context
, JSValueMakeNumber(context
, 1), exception
);
287 return JSValueToObject(context
, JSValueMakeNumber(context
, 0), exception
);
290 static bool MyObject_hasInstance(JSContextRef context
, JSObjectRef constructor
, JSValueRef possibleValue
, JSValueRef
* exception
)
292 UNUSED_PARAM(context
);
293 UNUSED_PARAM(constructor
);
295 if (JSValueIsString(context
, possibleValue
) && JSStringIsEqualToUTF8CString(JSValueToStringCopy(context
, possibleValue
, 0), "throwOnHasInstance")) {
296 JSEvaluateScript(context
, JSStringCreateWithUTF8CString("throw 'an exception'"), constructor
, JSStringCreateWithUTF8CString("test script"), 1, exception
);
300 JSStringRef numberString
= JSStringCreateWithUTF8CString("Number");
301 JSObjectRef numberConstructor
= JSValueToObject(context
, JSObjectGetProperty(context
, JSContextGetGlobalObject(context
), numberString
, exception
), exception
);
302 JSStringRelease(numberString
);
304 return JSValueIsInstanceOfConstructor(context
, possibleValue
, numberConstructor
, exception
);
307 static JSValueRef
MyObject_convertToType(JSContextRef context
, JSObjectRef object
, JSType type
, JSValueRef
* exception
)
309 UNUSED_PARAM(object
);
310 UNUSED_PARAM(exception
);
314 return JSValueMakeNumber(context
, 1);
317 JSStringRef string
= JSStringCreateWithUTF8CString("MyObjectAsString");
318 JSValueRef result
= JSValueMakeString(context
, string
);
319 JSStringRelease(string
);
326 // string conversion -- forward to default object class
327 return JSValueMakeNull(context
);
330 static JSValueRef
MyObject_convertToTypeWrapper(JSContextRef context
, JSObjectRef object
, JSType type
, JSValueRef
* exception
)
332 UNUSED_PARAM(context
);
333 UNUSED_PARAM(object
);
335 UNUSED_PARAM(exception
);
336 // Forward to default object class
340 static bool MyObject_set_nullGetForwardSet(JSContextRef ctx
, JSObjectRef object
, JSStringRef propertyName
, JSValueRef value
, JSValueRef
* exception
)
343 UNUSED_PARAM(object
);
344 UNUSED_PARAM(propertyName
);
346 UNUSED_PARAM(exception
);
347 return false; // Forward to parent class.
350 static JSStaticValue evilStaticValues
[] = {
351 { "nullGetSet", 0, 0, kJSPropertyAttributeNone
},
352 { "nullGetForwardSet", 0, MyObject_set_nullGetForwardSet
, kJSPropertyAttributeNone
},
356 static JSStaticFunction evilStaticFunctions
[] = {
357 { "nullCall", 0, kJSPropertyAttributeNone
},
361 JSClassDefinition MyObject_definition
= {
363 kJSClassAttributeNone
,
373 MyObject_hasProperty
,
374 MyObject_getProperty
,
375 MyObject_setProperty
,
376 MyObject_deleteProperty
,
377 MyObject_getPropertyNames
,
378 MyObject_callAsFunction
,
379 MyObject_callAsConstructor
,
380 MyObject_hasInstance
,
381 MyObject_convertToType
,
384 JSClassDefinition MyObject_convertToTypeWrapperDefinition
= {
386 kJSClassAttributeNone
,
404 MyObject_convertToTypeWrapper
,
407 JSClassDefinition MyObject_nullWrapperDefinition
= {
409 kJSClassAttributeNone
,
430 static JSClassRef
MyObject_class(JSContextRef context
)
432 UNUSED_PARAM(context
);
434 static JSClassRef jsClass
;
436 JSClassRef baseClass
= JSClassCreate(&MyObject_definition
);
437 MyObject_convertToTypeWrapperDefinition
.parentClass
= baseClass
;
438 JSClassRef wrapperClass
= JSClassCreate(&MyObject_convertToTypeWrapperDefinition
);
439 MyObject_nullWrapperDefinition
.parentClass
= wrapperClass
;
440 jsClass
= JSClassCreate(&MyObject_nullWrapperDefinition
);
446 static JSValueRef
PropertyCatchalls_getProperty(JSContextRef context
, JSObjectRef object
, JSStringRef propertyName
, JSValueRef
* exception
)
448 UNUSED_PARAM(context
);
449 UNUSED_PARAM(object
);
450 UNUSED_PARAM(propertyName
);
451 UNUSED_PARAM(exception
);
453 if (JSStringIsEqualToUTF8CString(propertyName
, "x")) {
458 // Swallow all .x gets after 5, returning null.
459 return JSValueMakeNull(context
);
462 if (JSStringIsEqualToUTF8CString(propertyName
, "y")) {
467 // Swallow all .y gets after 5, returning null.
468 return JSValueMakeNull(context
);
471 if (JSStringIsEqualToUTF8CString(propertyName
, "z")) {
476 // Swallow all .y gets after 5, returning null.
477 return JSValueMakeNull(context
);
483 static bool PropertyCatchalls_setProperty(JSContextRef context
, JSObjectRef object
, JSStringRef propertyName
, JSValueRef value
, JSValueRef
* exception
)
485 UNUSED_PARAM(context
);
486 UNUSED_PARAM(object
);
487 UNUSED_PARAM(propertyName
);
489 UNUSED_PARAM(exception
);
491 if (JSStringIsEqualToUTF8CString(propertyName
, "x")) {
496 // Swallow all .x sets after 4.
500 if (JSStringIsEqualToUTF8CString(propertyName
, "make_throw") || JSStringIsEqualToUTF8CString(propertyName
, "0")) {
501 *exception
= JSValueMakeNumber(context
, 5);
508 static void PropertyCatchalls_getPropertyNames(JSContextRef context
, JSObjectRef object
, JSPropertyNameAccumulatorRef propertyNames
)
510 UNUSED_PARAM(context
);
511 UNUSED_PARAM(object
);
514 static const char* numbers
[] = { "0", "1", "2", "3", "4", "5", "6", "7", "8", "9" };
516 // Provide a property of a different name every time.
517 JSStringRef propertyName
= JSStringCreateWithUTF8CString(numbers
[count
++ % 10]);
518 JSPropertyNameAccumulatorAddName(propertyNames
, propertyName
);
519 JSStringRelease(propertyName
);
522 JSClassDefinition PropertyCatchalls_definition
= {
524 kJSClassAttributeNone
,
535 PropertyCatchalls_getProperty
,
536 PropertyCatchalls_setProperty
,
538 PropertyCatchalls_getPropertyNames
,
545 static JSClassRef
PropertyCatchalls_class(JSContextRef context
)
547 UNUSED_PARAM(context
);
549 static JSClassRef jsClass
;
551 jsClass
= JSClassCreate(&PropertyCatchalls_definition
);
556 static bool EvilExceptionObject_hasInstance(JSContextRef context
, JSObjectRef constructor
, JSValueRef possibleValue
, JSValueRef
* exception
)
558 UNUSED_PARAM(context
);
559 UNUSED_PARAM(constructor
);
561 JSStringRef hasInstanceName
= JSStringCreateWithUTF8CString("hasInstance");
562 JSValueRef hasInstance
= JSObjectGetProperty(context
, constructor
, hasInstanceName
, exception
);
563 JSStringRelease(hasInstanceName
);
566 JSObjectRef function
= JSValueToObject(context
, hasInstance
, exception
);
567 JSValueRef result
= JSObjectCallAsFunction(context
, function
, constructor
, 1, &possibleValue
, exception
);
568 return result
&& JSValueToBoolean(context
, result
);
571 static JSValueRef
EvilExceptionObject_convertToType(JSContextRef context
, JSObjectRef object
, JSType type
, JSValueRef
* exception
)
573 UNUSED_PARAM(object
);
574 UNUSED_PARAM(exception
);
575 JSStringRef funcName
;
578 funcName
= JSStringCreateWithUTF8CString("toNumber");
581 funcName
= JSStringCreateWithUTF8CString("toStringExplicit");
584 return JSValueMakeNull(context
);
588 JSValueRef func
= JSObjectGetProperty(context
, object
, funcName
, exception
);
589 JSStringRelease(funcName
);
590 JSObjectRef function
= JSValueToObject(context
, func
, exception
);
592 return JSValueMakeNull(context
);
593 JSValueRef value
= JSObjectCallAsFunction(context
, function
, object
, 0, NULL
, exception
);
595 JSStringRef errorString
= JSStringCreateWithUTF8CString("convertToType failed");
596 JSValueRef errorStringRef
= JSValueMakeString(context
, errorString
);
597 JSStringRelease(errorString
);
598 return errorStringRef
;
603 JSClassDefinition EvilExceptionObject_definition
= {
605 kJSClassAttributeNone
,
607 "EvilExceptionObject",
622 EvilExceptionObject_hasInstance
,
623 EvilExceptionObject_convertToType
,
626 static JSClassRef
EvilExceptionObject_class(JSContextRef context
)
628 UNUSED_PARAM(context
);
630 static JSClassRef jsClass
;
632 jsClass
= JSClassCreate(&EvilExceptionObject_definition
);
637 JSClassDefinition EmptyObject_definition
= {
639 kJSClassAttributeNone
,
660 static JSClassRef
EmptyObject_class(JSContextRef context
)
662 UNUSED_PARAM(context
);
664 static JSClassRef jsClass
;
666 jsClass
= JSClassCreate(&EmptyObject_definition
);
672 static JSValueRef
Base_get(JSContextRef ctx
, JSObjectRef object
, JSStringRef propertyName
, JSValueRef
* exception
)
674 UNUSED_PARAM(object
);
675 UNUSED_PARAM(propertyName
);
676 UNUSED_PARAM(exception
);
678 return JSValueMakeNumber(ctx
, 1); // distinguish base get form derived get
681 static bool Base_set(JSContextRef ctx
, JSObjectRef object
, JSStringRef propertyName
, JSValueRef value
, JSValueRef
* exception
)
683 UNUSED_PARAM(object
);
684 UNUSED_PARAM(propertyName
);
687 *exception
= JSValueMakeNumber(ctx
, 1); // distinguish base set from derived set
691 static JSValueRef
Base_callAsFunction(JSContextRef ctx
, JSObjectRef function
, JSObjectRef thisObject
, size_t argumentCount
, const JSValueRef arguments
[], JSValueRef
* exception
)
693 UNUSED_PARAM(function
);
694 UNUSED_PARAM(thisObject
);
695 UNUSED_PARAM(argumentCount
);
696 UNUSED_PARAM(arguments
);
697 UNUSED_PARAM(exception
);
699 return JSValueMakeNumber(ctx
, 1); // distinguish base call from derived call
702 static JSValueRef
Base_returnHardNull(JSContextRef ctx
, JSObjectRef function
, JSObjectRef thisObject
, size_t argumentCount
, const JSValueRef arguments
[], JSValueRef
* exception
)
705 UNUSED_PARAM(function
);
706 UNUSED_PARAM(thisObject
);
707 UNUSED_PARAM(argumentCount
);
708 UNUSED_PARAM(arguments
);
709 UNUSED_PARAM(exception
);
711 return 0; // should convert to undefined!
714 static JSStaticFunction Base_staticFunctions
[] = {
715 { "baseProtoDup", NULL
, kJSPropertyAttributeNone
},
716 { "baseProto", Base_callAsFunction
, kJSPropertyAttributeNone
},
717 { "baseHardNull", Base_returnHardNull
, kJSPropertyAttributeNone
},
721 static JSStaticValue Base_staticValues
[] = {
722 { "baseDup", Base_get
, Base_set
, kJSPropertyAttributeNone
},
723 { "baseOnly", Base_get
, Base_set
, kJSPropertyAttributeNone
},
727 static bool TestInitializeFinalize
;
728 static void Base_initialize(JSContextRef context
, JSObjectRef object
)
730 UNUSED_PARAM(context
);
732 if (TestInitializeFinalize
) {
733 ASSERT((void*)1 == JSObjectGetPrivate(object
));
734 JSObjectSetPrivate(object
, (void*)2);
738 static unsigned Base_didFinalize
;
739 static void Base_finalize(JSObjectRef object
)
741 UNUSED_PARAM(object
);
742 if (TestInitializeFinalize
) {
743 ASSERT((void*)4 == JSObjectGetPrivate(object
));
744 Base_didFinalize
= true;
748 static JSClassRef
Base_class(JSContextRef context
)
750 UNUSED_PARAM(context
);
752 static JSClassRef jsClass
;
754 JSClassDefinition definition
= kJSClassDefinitionEmpty
;
755 definition
.staticValues
= Base_staticValues
;
756 definition
.staticFunctions
= Base_staticFunctions
;
757 definition
.initialize
= Base_initialize
;
758 definition
.finalize
= Base_finalize
;
759 jsClass
= JSClassCreate(&definition
);
764 static JSValueRef
Derived_get(JSContextRef ctx
, JSObjectRef object
, JSStringRef propertyName
, JSValueRef
* exception
)
766 UNUSED_PARAM(object
);
767 UNUSED_PARAM(propertyName
);
768 UNUSED_PARAM(exception
);
770 return JSValueMakeNumber(ctx
, 2); // distinguish base get form derived get
773 static bool Derived_set(JSContextRef ctx
, JSObjectRef object
, JSStringRef propertyName
, JSValueRef value
, JSValueRef
* exception
)
776 UNUSED_PARAM(object
);
777 UNUSED_PARAM(propertyName
);
780 *exception
= JSValueMakeNumber(ctx
, 2); // distinguish base set from derived set
784 static JSValueRef
Derived_callAsFunction(JSContextRef ctx
, JSObjectRef function
, JSObjectRef thisObject
, size_t argumentCount
, const JSValueRef arguments
[], JSValueRef
* exception
)
786 UNUSED_PARAM(function
);
787 UNUSED_PARAM(thisObject
);
788 UNUSED_PARAM(argumentCount
);
789 UNUSED_PARAM(arguments
);
790 UNUSED_PARAM(exception
);
792 return JSValueMakeNumber(ctx
, 2); // distinguish base call from derived call
795 static JSStaticFunction Derived_staticFunctions
[] = {
796 { "protoOnly", Derived_callAsFunction
, kJSPropertyAttributeNone
},
797 { "protoDup", NULL
, kJSPropertyAttributeNone
},
798 { "baseProtoDup", Derived_callAsFunction
, kJSPropertyAttributeNone
},
802 static JSStaticValue Derived_staticValues
[] = {
803 { "derivedOnly", Derived_get
, Derived_set
, kJSPropertyAttributeNone
},
804 { "protoDup", Derived_get
, Derived_set
, kJSPropertyAttributeNone
},
805 { "baseDup", Derived_get
, Derived_set
, kJSPropertyAttributeNone
},
809 static void Derived_initialize(JSContextRef context
, JSObjectRef object
)
811 UNUSED_PARAM(context
);
813 if (TestInitializeFinalize
) {
814 ASSERT((void*)2 == JSObjectGetPrivate(object
));
815 JSObjectSetPrivate(object
, (void*)3);
819 static void Derived_finalize(JSObjectRef object
)
821 if (TestInitializeFinalize
) {
822 ASSERT((void*)3 == JSObjectGetPrivate(object
));
823 JSObjectSetPrivate(object
, (void*)4);
827 static JSClassRef
Derived_class(JSContextRef context
)
829 static JSClassRef jsClass
;
831 JSClassDefinition definition
= kJSClassDefinitionEmpty
;
832 definition
.parentClass
= Base_class(context
);
833 definition
.staticValues
= Derived_staticValues
;
834 definition
.staticFunctions
= Derived_staticFunctions
;
835 definition
.initialize
= Derived_initialize
;
836 definition
.finalize
= Derived_finalize
;
837 jsClass
= JSClassCreate(&definition
);
842 static JSClassRef
Derived2_class(JSContextRef context
)
844 static JSClassRef jsClass
;
846 JSClassDefinition definition
= kJSClassDefinitionEmpty
;
847 definition
.parentClass
= Derived_class(context
);
848 jsClass
= JSClassCreate(&definition
);
853 static JSValueRef
print_callAsFunction(JSContextRef ctx
, JSObjectRef functionObject
, JSObjectRef thisObject
, size_t argumentCount
, const JSValueRef arguments
[], JSValueRef
* exception
)
855 UNUSED_PARAM(functionObject
);
856 UNUSED_PARAM(thisObject
);
857 UNUSED_PARAM(exception
);
859 ASSERT(JSContextGetGlobalContext(ctx
) == context
);
861 if (argumentCount
> 0) {
862 JSStringRef string
= JSValueToStringCopy(ctx
, arguments
[0], NULL
);
863 size_t sizeUTF8
= JSStringGetMaximumUTF8CStringSize(string
);
864 char* stringUTF8
= (char*)malloc(sizeUTF8
);
865 JSStringGetUTF8CString(string
, stringUTF8
, sizeUTF8
);
866 printf("%s\n", stringUTF8
);
868 JSStringRelease(string
);
871 return JSValueMakeUndefined(ctx
);
874 static JSObjectRef
myConstructor_callAsConstructor(JSContextRef context
, JSObjectRef constructorObject
, size_t argumentCount
, const JSValueRef arguments
[], JSValueRef
* exception
)
876 UNUSED_PARAM(constructorObject
);
877 UNUSED_PARAM(exception
);
879 JSObjectRef result
= JSObjectMake(context
, NULL
, NULL
);
880 if (argumentCount
> 0) {
881 JSStringRef value
= JSStringCreateWithUTF8CString("value");
882 JSObjectSetProperty(context
, result
, value
, arguments
[0], kJSPropertyAttributeNone
, NULL
);
883 JSStringRelease(value
);
889 static JSObjectRef
myBadConstructor_callAsConstructor(JSContextRef context
, JSObjectRef constructorObject
, size_t argumentCount
, const JSValueRef arguments
[], JSValueRef
* exception
)
891 UNUSED_PARAM(context
);
892 UNUSED_PARAM(constructorObject
);
893 UNUSED_PARAM(argumentCount
);
894 UNUSED_PARAM(arguments
);
895 UNUSED_PARAM(exception
);
901 static void globalObject_initialize(JSContextRef context
, JSObjectRef object
)
903 UNUSED_PARAM(object
);
904 // Ensure that an execution context is passed in
907 // Ensure that the global object is set to the object that we were passed
908 JSObjectRef globalObject
= JSContextGetGlobalObject(context
);
909 ASSERT(globalObject
);
910 ASSERT(object
== globalObject
);
912 // Ensure that the standard global properties have been set on the global object
913 JSStringRef array
= JSStringCreateWithUTF8CString("Array");
914 JSObjectRef arrayConstructor
= JSValueToObject(context
, JSObjectGetProperty(context
, globalObject
, array
, NULL
), NULL
);
915 JSStringRelease(array
);
917 UNUSED_PARAM(arrayConstructor
);
918 ASSERT(arrayConstructor
);
921 static JSValueRef
globalObject_get(JSContextRef ctx
, JSObjectRef object
, JSStringRef propertyName
, JSValueRef
* exception
)
923 UNUSED_PARAM(object
);
924 UNUSED_PARAM(propertyName
);
925 UNUSED_PARAM(exception
);
927 return JSValueMakeNumber(ctx
, 3);
930 static bool globalObject_set(JSContextRef ctx
, JSObjectRef object
, JSStringRef propertyName
, JSValueRef value
, JSValueRef
* exception
)
932 UNUSED_PARAM(object
);
933 UNUSED_PARAM(propertyName
);
936 *exception
= JSValueMakeNumber(ctx
, 3);
940 static JSValueRef
globalObject_call(JSContextRef ctx
, JSObjectRef function
, JSObjectRef thisObject
, size_t argumentCount
, const JSValueRef arguments
[], JSValueRef
* exception
)
942 UNUSED_PARAM(function
);
943 UNUSED_PARAM(thisObject
);
944 UNUSED_PARAM(argumentCount
);
945 UNUSED_PARAM(arguments
);
946 UNUSED_PARAM(exception
);
948 return JSValueMakeNumber(ctx
, 3);
951 static JSValueRef
functionGC(JSContextRef context
, JSObjectRef function
, JSObjectRef thisObject
, size_t argumentCount
, const JSValueRef arguments
[], JSValueRef
* exception
)
953 UNUSED_PARAM(function
);
954 UNUSED_PARAM(thisObject
);
955 UNUSED_PARAM(argumentCount
);
956 UNUSED_PARAM(arguments
);
957 UNUSED_PARAM(exception
);
958 JSGarbageCollect(context
);
959 return JSValueMakeUndefined(context
);
962 static JSStaticValue globalObject_staticValues
[] = {
963 { "globalStaticValue", globalObject_get
, globalObject_set
, kJSPropertyAttributeNone
},
967 static JSStaticFunction globalObject_staticFunctions
[] = {
968 { "globalStaticFunction", globalObject_call
, kJSPropertyAttributeNone
},
969 { "gc", functionGC
, kJSPropertyAttributeNone
},
973 static char* createStringWithContentsOfFile(const char* fileName
);
975 static void testInitializeFinalize()
977 JSObjectRef o
= JSObjectMake(context
, Derived_class(context
), (void*)1);
979 ASSERT(JSObjectGetPrivate(o
) == (void*)3);
982 static JSValueRef jsNumberValue
= NULL
;
984 static JSObjectRef aHeapRef
= NULL
;
986 static void makeGlobalNumberValue(JSContextRef context
) {
987 JSValueRef v
= JSValueMakeNumber(context
, 420);
988 JSValueProtect(context
, v
);
993 static bool assertTrue(bool value
, const char* message
)
997 fprintf(stderr
, "assertTrue failed: '%s'\n", message
);
999 fprintf(stderr
, "assertTrue failed.\n");
1005 static bool checkForCycleInPrototypeChain()
1008 JSGlobalContextRef context
= JSGlobalContextCreate(0);
1009 JSObjectRef object1
= JSObjectMake(context
, /* jsClass */ 0, /* data */ 0);
1010 JSObjectRef object2
= JSObjectMake(context
, /* jsClass */ 0, /* data */ 0);
1011 JSObjectRef object3
= JSObjectMake(context
, /* jsClass */ 0, /* data */ 0);
1013 JSObjectSetPrototype(context
, object1
, JSValueMakeNull(context
));
1014 ASSERT(JSValueIsNull(context
, JSObjectGetPrototype(context
, object1
)));
1016 // object1 -> object1
1017 JSObjectSetPrototype(context
, object1
, object1
);
1018 result
&= assertTrue(JSValueIsNull(context
, JSObjectGetPrototype(context
, object1
)), "It is possible to assign self as a prototype");
1020 // object1 -> object2 -> object1
1021 JSObjectSetPrototype(context
, object2
, object1
);
1022 ASSERT(JSValueIsStrictEqual(context
, JSObjectGetPrototype(context
, object2
), object1
));
1023 JSObjectSetPrototype(context
, object1
, object2
);
1024 result
&= assertTrue(JSValueIsNull(context
, JSObjectGetPrototype(context
, object1
)), "It is possible to close a prototype chain cycle");
1026 // object1 -> object2 -> object3 -> object1
1027 JSObjectSetPrototype(context
, object2
, object3
);
1028 ASSERT(JSValueIsStrictEqual(context
, JSObjectGetPrototype(context
, object2
), object3
));
1029 JSObjectSetPrototype(context
, object1
, object2
);
1030 ASSERT(JSValueIsStrictEqual(context
, JSObjectGetPrototype(context
, object1
), object2
));
1031 JSObjectSetPrototype(context
, object3
, object1
);
1032 result
&= assertTrue(!JSValueIsStrictEqual(context
, JSObjectGetPrototype(context
, object3
), object1
), "It is possible to close a prototype chain cycle");
1034 JSValueRef exception
;
1035 JSStringRef code
= JSStringCreateWithUTF8CString("o = { }; p = { }; o.__proto__ = p; p.__proto__ = o");
1036 JSStringRef file
= JSStringCreateWithUTF8CString("");
1037 result
&= assertTrue(!JSEvaluateScript(context
, code
, /* thisObject*/ 0, file
, 1, &exception
)
1038 , "An exception should be thrown");
1040 JSStringRelease(code
);
1041 JSStringRelease(file
);
1042 JSGlobalContextRelease(context
);
1046 static void checkConstnessInJSObjectNames()
1048 JSStaticFunction fun
;
1049 fun
.name
= "something";
1051 val
.name
= "something";
1054 #if PLATFORM(MAC) || PLATFORM(IOS)
1055 static double currentCPUTime()
1057 mach_msg_type_number_t infoCount
= THREAD_BASIC_INFO_COUNT
;
1058 thread_basic_info_data_t info
;
1060 /* Get thread information */
1061 mach_port_t threadPort
= mach_thread_self();
1062 thread_info(threadPort
, THREAD_BASIC_INFO
, (thread_info_t
)(&info
), &infoCount
);
1063 mach_port_deallocate(mach_task_self(), threadPort
);
1065 double time
= info
.user_time
.seconds
+ info
.user_time
.microseconds
/ 1000000.;
1066 time
+= info
.system_time
.seconds
+ info
.system_time
.microseconds
/ 1000000.;
1071 static JSValueRef
currentCPUTime_callAsFunction(JSContextRef ctx
, JSObjectRef functionObject
, JSObjectRef thisObject
, size_t argumentCount
, const JSValueRef arguments
[], JSValueRef
* exception
)
1073 UNUSED_PARAM(functionObject
);
1074 UNUSED_PARAM(thisObject
);
1075 UNUSED_PARAM(argumentCount
);
1076 UNUSED_PARAM(arguments
);
1077 UNUSED_PARAM(exception
);
1079 ASSERT(JSContextGetGlobalContext(ctx
) == context
);
1080 return JSValueMakeNumber(ctx
, currentCPUTime());
1083 bool shouldTerminateCallbackWasCalled
= false;
1084 static bool shouldTerminateCallback(JSContextRef ctx
, void* context
)
1087 UNUSED_PARAM(context
);
1088 shouldTerminateCallbackWasCalled
= true;
1092 bool cancelTerminateCallbackWasCalled
= false;
1093 static bool cancelTerminateCallback(JSContextRef ctx
, void* context
)
1096 UNUSED_PARAM(context
);
1097 cancelTerminateCallbackWasCalled
= true;
1101 int extendTerminateCallbackCalled
= 0;
1102 static bool extendTerminateCallback(JSContextRef ctx
, void* context
)
1104 UNUSED_PARAM(context
);
1105 extendTerminateCallbackCalled
++;
1106 if (extendTerminateCallbackCalled
== 1) {
1107 JSContextGroupRef contextGroup
= JSContextGetGroup(ctx
);
1108 JSContextGroupSetExecutionTimeLimit(contextGroup
, .200f
, extendTerminateCallback
, 0);
1113 #endif /* PLATFORM(MAC) || PLATFORM(IOS) */
1116 int main(int argc
, char* argv
[])
1119 // Cygwin calls ::SetErrorMode(SEM_FAILCRITICALERRORS), which we will inherit. This is bad for
1120 // testing/debugging, as it causes the post-mortem debugger not to be invoked. We reset the
1121 // error mode here to work around Cygwin's behavior. See <http://webkit.org/b/55222>.
1125 #if JSC_OBJC_API_ENABLED
1126 testObjectiveCAPI();
1129 const char *scriptPath
= "testapi.js";
1131 scriptPath
= argv
[1];
1134 // Test garbage collection with a fresh context
1135 context
= JSGlobalContextCreateInGroup(NULL
, NULL
);
1136 TestInitializeFinalize
= true;
1137 testInitializeFinalize();
1138 JSGlobalContextRelease(context
);
1139 TestInitializeFinalize
= false;
1141 ASSERT(Base_didFinalize
);
1143 JSClassDefinition globalObjectClassDefinition
= kJSClassDefinitionEmpty
;
1144 globalObjectClassDefinition
.initialize
= globalObject_initialize
;
1145 globalObjectClassDefinition
.staticValues
= globalObject_staticValues
;
1146 globalObjectClassDefinition
.staticFunctions
= globalObject_staticFunctions
;
1147 globalObjectClassDefinition
.attributes
= kJSClassAttributeNoAutomaticPrototype
;
1148 JSClassRef globalObjectClass
= JSClassCreate(&globalObjectClassDefinition
);
1149 context
= JSGlobalContextCreateInGroup(NULL
, globalObjectClass
);
1151 JSContextGroupRef contextGroup
= JSContextGetGroup(context
);
1153 JSGlobalContextRetain(context
);
1154 JSGlobalContextRelease(context
);
1155 ASSERT(JSContextGetGlobalContext(context
) == context
);
1157 JSReportExtraMemoryCost(context
, 0);
1158 JSReportExtraMemoryCost(context
, 1);
1159 JSReportExtraMemoryCost(context
, 1024);
1161 JSObjectRef globalObject
= JSContextGetGlobalObject(context
);
1162 ASSERT(JSValueIsObject(context
, globalObject
));
1164 JSValueRef jsUndefined
= JSValueMakeUndefined(context
);
1165 JSValueRef jsNull
= JSValueMakeNull(context
);
1166 JSValueRef jsTrue
= JSValueMakeBoolean(context
, true);
1167 JSValueRef jsFalse
= JSValueMakeBoolean(context
, false);
1168 JSValueRef jsZero
= JSValueMakeNumber(context
, 0);
1169 JSValueRef jsOne
= JSValueMakeNumber(context
, 1);
1170 JSValueRef jsOneThird
= JSValueMakeNumber(context
, 1.0 / 3.0);
1171 JSObjectRef jsObjectNoProto
= JSObjectMake(context
, NULL
, NULL
);
1172 JSObjectSetPrototype(context
, jsObjectNoProto
, JSValueMakeNull(context
));
1174 // FIXME: test funny utf8 characters
1175 JSStringRef jsEmptyIString
= JSStringCreateWithUTF8CString("");
1176 JSValueRef jsEmptyString
= JSValueMakeString(context
, jsEmptyIString
);
1178 JSStringRef jsOneIString
= JSStringCreateWithUTF8CString("1");
1179 JSValueRef jsOneString
= JSValueMakeString(context
, jsOneIString
);
1181 UniChar singleUniChar
= 65; // Capital A
1182 CFMutableStringRef cfString
=
1183 CFStringCreateMutableWithExternalCharactersNoCopy(kCFAllocatorDefault
,
1189 JSStringRef jsCFIString
= JSStringCreateWithCFString(cfString
);
1190 JSValueRef jsCFString
= JSValueMakeString(context
, jsCFIString
);
1192 CFStringRef cfEmptyString
= CFStringCreateWithCString(kCFAllocatorDefault
, "", kCFStringEncodingUTF8
);
1194 JSStringRef jsCFEmptyIString
= JSStringCreateWithCFString(cfEmptyString
);
1195 JSValueRef jsCFEmptyString
= JSValueMakeString(context
, jsCFEmptyIString
);
1197 CFIndex cfStringLength
= CFStringGetLength(cfString
);
1198 UniChar
* buffer
= (UniChar
*)malloc(cfStringLength
* sizeof(UniChar
));
1199 CFStringGetCharacters(cfString
,
1200 CFRangeMake(0, cfStringLength
),
1202 JSStringRef jsCFIStringWithCharacters
= JSStringCreateWithCharacters((JSChar
*)buffer
, cfStringLength
);
1203 JSValueRef jsCFStringWithCharacters
= JSValueMakeString(context
, jsCFIStringWithCharacters
);
1205 JSStringRef jsCFEmptyIStringWithCharacters
= JSStringCreateWithCharacters((JSChar
*)buffer
, CFStringGetLength(cfEmptyString
));
1207 JSValueRef jsCFEmptyStringWithCharacters
= JSValueMakeString(context
, jsCFEmptyIStringWithCharacters
);
1209 JSChar constantString
[] = { 'H', 'e', 'l', 'l', 'o', };
1210 JSStringRef constantStringRef
= JSStringCreateWithCharactersNoCopy(constantString
, sizeof(constantString
) / sizeof(constantString
[0]));
1211 ASSERT(JSStringGetCharactersPtr(constantStringRef
) == constantString
);
1212 JSStringRelease(constantStringRef
);
1214 ASSERT(JSValueGetType(context
, NULL
) == kJSTypeNull
);
1215 ASSERT(JSValueGetType(context
, jsUndefined
) == kJSTypeUndefined
);
1216 ASSERT(JSValueGetType(context
, jsNull
) == kJSTypeNull
);
1217 ASSERT(JSValueGetType(context
, jsTrue
) == kJSTypeBoolean
);
1218 ASSERT(JSValueGetType(context
, jsFalse
) == kJSTypeBoolean
);
1219 ASSERT(JSValueGetType(context
, jsZero
) == kJSTypeNumber
);
1220 ASSERT(JSValueGetType(context
, jsOne
) == kJSTypeNumber
);
1221 ASSERT(JSValueGetType(context
, jsOneThird
) == kJSTypeNumber
);
1222 ASSERT(JSValueGetType(context
, jsEmptyString
) == kJSTypeString
);
1223 ASSERT(JSValueGetType(context
, jsOneString
) == kJSTypeString
);
1224 ASSERT(JSValueGetType(context
, jsCFString
) == kJSTypeString
);
1225 ASSERT(JSValueGetType(context
, jsCFStringWithCharacters
) == kJSTypeString
);
1226 ASSERT(JSValueGetType(context
, jsCFEmptyString
) == kJSTypeString
);
1227 ASSERT(JSValueGetType(context
, jsCFEmptyStringWithCharacters
) == kJSTypeString
);
1229 ASSERT(!JSValueIsBoolean(context
, NULL
));
1230 ASSERT(!JSValueIsObject(context
, NULL
));
1231 ASSERT(!JSValueIsString(context
, NULL
));
1232 ASSERT(!JSValueIsNumber(context
, NULL
));
1233 ASSERT(!JSValueIsUndefined(context
, NULL
));
1234 ASSERT(JSValueIsNull(context
, NULL
));
1235 ASSERT(!JSObjectCallAsFunction(context
, NULL
, NULL
, 0, NULL
, NULL
));
1236 ASSERT(!JSObjectCallAsConstructor(context
, NULL
, 0, NULL
, NULL
));
1237 ASSERT(!JSObjectIsConstructor(context
, NULL
));
1238 ASSERT(!JSObjectIsFunction(context
, NULL
));
1240 JSStringRef nullString
= JSStringCreateWithUTF8CString(0);
1241 const JSChar
* characters
= JSStringGetCharactersPtr(nullString
);
1243 printf("FAIL: Didn't return null when accessing character pointer of a null String.\n");
1246 printf("PASS: returned null when accessing character pointer of a null String.\n");
1248 size_t length
= JSStringGetLength(nullString
);
1250 printf("FAIL: Didn't return 0 length for null String.\n");
1253 printf("PASS: returned 0 length for null String.\n");
1254 JSStringRelease(nullString
);
1256 JSObjectRef propertyCatchalls
= JSObjectMake(context
, PropertyCatchalls_class(context
), NULL
);
1257 JSStringRef propertyCatchallsString
= JSStringCreateWithUTF8CString("PropertyCatchalls");
1258 JSObjectSetProperty(context
, globalObject
, propertyCatchallsString
, propertyCatchalls
, kJSPropertyAttributeNone
, NULL
);
1259 JSStringRelease(propertyCatchallsString
);
1261 JSObjectRef myObject
= JSObjectMake(context
, MyObject_class(context
), NULL
);
1262 JSStringRef myObjectIString
= JSStringCreateWithUTF8CString("MyObject");
1263 JSObjectSetProperty(context
, globalObject
, myObjectIString
, myObject
, kJSPropertyAttributeNone
, NULL
);
1264 JSStringRelease(myObjectIString
);
1266 JSObjectRef EvilExceptionObject
= JSObjectMake(context
, EvilExceptionObject_class(context
), NULL
);
1267 JSStringRef EvilExceptionObjectIString
= JSStringCreateWithUTF8CString("EvilExceptionObject");
1268 JSObjectSetProperty(context
, globalObject
, EvilExceptionObjectIString
, EvilExceptionObject
, kJSPropertyAttributeNone
, NULL
);
1269 JSStringRelease(EvilExceptionObjectIString
);
1271 JSObjectRef EmptyObject
= JSObjectMake(context
, EmptyObject_class(context
), NULL
);
1272 JSStringRef EmptyObjectIString
= JSStringCreateWithUTF8CString("EmptyObject");
1273 JSObjectSetProperty(context
, globalObject
, EmptyObjectIString
, EmptyObject
, kJSPropertyAttributeNone
, NULL
);
1274 JSStringRelease(EmptyObjectIString
);
1276 JSStringRef lengthStr
= JSStringCreateWithUTF8CString("length");
1277 JSObjectRef aStackRef
= JSObjectMakeArray(context
, 0, 0, 0);
1278 aHeapRef
= aStackRef
;
1279 JSObjectSetProperty(context
, aHeapRef
, lengthStr
, JSValueMakeNumber(context
, 10), 0, 0);
1280 JSStringRef privatePropertyName
= JSStringCreateWithUTF8CString("privateProperty");
1281 if (!JSObjectSetPrivateProperty(context
, myObject
, privatePropertyName
, aHeapRef
)) {
1282 printf("FAIL: Could not set private property.\n");
1285 printf("PASS: Set private property.\n");
1287 if (JSObjectSetPrivateProperty(context
, aHeapRef
, privatePropertyName
, aHeapRef
)) {
1288 printf("FAIL: JSObjectSetPrivateProperty should fail on non-API objects.\n");
1291 printf("PASS: Did not allow JSObjectSetPrivateProperty on a non-API object.\n");
1292 if (JSObjectGetPrivateProperty(context
, myObject
, privatePropertyName
) != aHeapRef
) {
1293 printf("FAIL: Could not retrieve private property.\n");
1296 printf("PASS: Retrieved private property.\n");
1297 if (JSObjectGetPrivateProperty(context
, aHeapRef
, privatePropertyName
)) {
1298 printf("FAIL: JSObjectGetPrivateProperty should return NULL when called on a non-API object.\n");
1301 printf("PASS: JSObjectGetPrivateProperty return NULL.\n");
1303 if (JSObjectGetProperty(context
, myObject
, privatePropertyName
, 0) == aHeapRef
) {
1304 printf("FAIL: Accessed private property through ordinary property lookup.\n");
1307 printf("PASS: Cannot access private property through ordinary property lookup.\n");
1309 JSGarbageCollect(context
);
1311 for (int i
= 0; i
< 10000; i
++)
1312 JSObjectMake(context
, 0, 0);
1314 aHeapRef
= JSValueToObject(context
, JSObjectGetPrivateProperty(context
, myObject
, privatePropertyName
), 0);
1315 if (JSValueToNumber(context
, JSObjectGetProperty(context
, aHeapRef
, lengthStr
, 0), 0) != 10) {
1316 printf("FAIL: Private property has been collected.\n");
1319 printf("PASS: Private property does not appear to have been collected.\n");
1320 JSStringRelease(lengthStr
);
1322 if (!JSObjectSetPrivateProperty(context
, myObject
, privatePropertyName
, 0)) {
1323 printf("FAIL: Could not set private property to NULL.\n");
1326 printf("PASS: Set private property to NULL.\n");
1327 if (JSObjectGetPrivateProperty(context
, myObject
, privatePropertyName
)) {
1328 printf("FAIL: Could not retrieve private property.\n");
1331 printf("PASS: Retrieved private property.\n");
1333 JSStringRef nullJSON
= JSStringCreateWithUTF8CString(0);
1334 JSValueRef nullJSONObject
= JSValueMakeFromJSONString(context
, nullJSON
);
1335 if (nullJSONObject
) {
1336 printf("FAIL: Did not parse null String as JSON correctly\n");
1339 printf("PASS: Parsed null String as JSON correctly.\n");
1340 JSStringRelease(nullJSON
);
1342 JSStringRef validJSON
= JSStringCreateWithUTF8CString("{\"aProperty\":true}");
1343 JSValueRef jsonObject
= JSValueMakeFromJSONString(context
, validJSON
);
1344 JSStringRelease(validJSON
);
1345 if (!JSValueIsObject(context
, jsonObject
)) {
1346 printf("FAIL: Did not parse valid JSON correctly\n");
1349 printf("PASS: Parsed valid JSON string.\n");
1350 JSStringRef propertyName
= JSStringCreateWithUTF8CString("aProperty");
1351 assertEqualsAsBoolean(JSObjectGetProperty(context
, JSValueToObject(context
, jsonObject
, 0), propertyName
, 0), true);
1352 JSStringRelease(propertyName
);
1353 JSStringRef invalidJSON
= JSStringCreateWithUTF8CString("fail!");
1354 if (JSValueMakeFromJSONString(context
, invalidJSON
)) {
1355 printf("FAIL: Should return null for invalid JSON data\n");
1358 printf("PASS: Correctly returned null for invalid JSON data.\n");
1359 JSValueRef exception
;
1360 JSStringRef str
= JSValueCreateJSONString(context
, jsonObject
, 0, 0);
1361 if (!JSStringIsEqualToUTF8CString(str
, "{\"aProperty\":true}")) {
1362 printf("FAIL: Did not correctly serialise with indent of 0.\n");
1365 printf("PASS: Correctly serialised with indent of 0.\n");
1366 JSStringRelease(str
);
1368 str
= JSValueCreateJSONString(context
, jsonObject
, 4, 0);
1369 if (!JSStringIsEqualToUTF8CString(str
, "{\n \"aProperty\": true\n}")) {
1370 printf("FAIL: Did not correctly serialise with indent of 4.\n");
1373 printf("PASS: Correctly serialised with indent of 4.\n");
1374 JSStringRelease(str
);
1375 JSStringRef src
= JSStringCreateWithUTF8CString("({get a(){ throw '';}})");
1376 JSValueRef unstringifiableObj
= JSEvaluateScript(context
, src
, NULL
, NULL
, 1, NULL
);
1378 str
= JSValueCreateJSONString(context
, unstringifiableObj
, 4, 0);
1380 printf("FAIL: Didn't return null when attempting to serialize unserializable value.\n");
1381 JSStringRelease(str
);
1384 printf("PASS: returned null when attempting to serialize unserializable value.\n");
1386 str
= JSValueCreateJSONString(context
, unstringifiableObj
, 4, &exception
);
1388 printf("FAIL: Didn't return null when attempting to serialize unserializable value.\n");
1389 JSStringRelease(str
);
1392 printf("PASS: returned null when attempting to serialize unserializable value.\n");
1394 printf("FAIL: Did not set exception on serialisation error\n");
1397 printf("PASS: set exception on serialisation error\n");
1398 // Conversions that throw exceptions
1400 ASSERT(NULL
== JSValueToObject(context
, jsNull
, &exception
));
1404 // FIXME <rdar://4668451> - On i386 the isnan(double) macro tries to map to the isnan(float) function,
1405 // causing a build break with -Wshorten-64-to-32 enabled. The issue is known by the appropriate team.
1406 // After that's resolved, we can remove these casts
1407 ASSERT(isnan((float)JSValueToNumber(context
, jsObjectNoProto
, &exception
)));
1411 ASSERT(!JSValueToStringCopy(context
, jsObjectNoProto
, &exception
));
1414 ASSERT(JSValueToBoolean(context
, myObject
));
1417 ASSERT(!JSValueIsEqual(context
, jsObjectNoProto
, JSValueMakeNumber(context
, 1), &exception
));
1421 JSObjectGetPropertyAtIndex(context
, myObject
, 0, &exception
);
1422 ASSERT(1 == JSValueToNumber(context
, exception
, NULL
));
1424 assertEqualsAsBoolean(jsUndefined
, false);
1425 assertEqualsAsBoolean(jsNull
, false);
1426 assertEqualsAsBoolean(jsTrue
, true);
1427 assertEqualsAsBoolean(jsFalse
, false);
1428 assertEqualsAsBoolean(jsZero
, false);
1429 assertEqualsAsBoolean(jsOne
, true);
1430 assertEqualsAsBoolean(jsOneThird
, true);
1431 assertEqualsAsBoolean(jsEmptyString
, false);
1432 assertEqualsAsBoolean(jsOneString
, true);
1433 assertEqualsAsBoolean(jsCFString
, true);
1434 assertEqualsAsBoolean(jsCFStringWithCharacters
, true);
1435 assertEqualsAsBoolean(jsCFEmptyString
, false);
1436 assertEqualsAsBoolean(jsCFEmptyStringWithCharacters
, false);
1438 assertEqualsAsNumber(jsUndefined
, nan(""));
1439 assertEqualsAsNumber(jsNull
, 0);
1440 assertEqualsAsNumber(jsTrue
, 1);
1441 assertEqualsAsNumber(jsFalse
, 0);
1442 assertEqualsAsNumber(jsZero
, 0);
1443 assertEqualsAsNumber(jsOne
, 1);
1444 assertEqualsAsNumber(jsOneThird
, 1.0 / 3.0);
1445 assertEqualsAsNumber(jsEmptyString
, 0);
1446 assertEqualsAsNumber(jsOneString
, 1);
1447 assertEqualsAsNumber(jsCFString
, nan(""));
1448 assertEqualsAsNumber(jsCFStringWithCharacters
, nan(""));
1449 assertEqualsAsNumber(jsCFEmptyString
, 0);
1450 assertEqualsAsNumber(jsCFEmptyStringWithCharacters
, 0);
1451 ASSERT(sizeof(JSChar
) == sizeof(UniChar
));
1453 assertEqualsAsCharactersPtr(jsUndefined
, "undefined");
1454 assertEqualsAsCharactersPtr(jsNull
, "null");
1455 assertEqualsAsCharactersPtr(jsTrue
, "true");
1456 assertEqualsAsCharactersPtr(jsFalse
, "false");
1457 assertEqualsAsCharactersPtr(jsZero
, "0");
1458 assertEqualsAsCharactersPtr(jsOne
, "1");
1459 assertEqualsAsCharactersPtr(jsOneThird
, "0.3333333333333333");
1460 assertEqualsAsCharactersPtr(jsEmptyString
, "");
1461 assertEqualsAsCharactersPtr(jsOneString
, "1");
1462 assertEqualsAsCharactersPtr(jsCFString
, "A");
1463 assertEqualsAsCharactersPtr(jsCFStringWithCharacters
, "A");
1464 assertEqualsAsCharactersPtr(jsCFEmptyString
, "");
1465 assertEqualsAsCharactersPtr(jsCFEmptyStringWithCharacters
, "");
1467 assertEqualsAsUTF8String(jsUndefined
, "undefined");
1468 assertEqualsAsUTF8String(jsNull
, "null");
1469 assertEqualsAsUTF8String(jsTrue
, "true");
1470 assertEqualsAsUTF8String(jsFalse
, "false");
1471 assertEqualsAsUTF8String(jsZero
, "0");
1472 assertEqualsAsUTF8String(jsOne
, "1");
1473 assertEqualsAsUTF8String(jsOneThird
, "0.3333333333333333");
1474 assertEqualsAsUTF8String(jsEmptyString
, "");
1475 assertEqualsAsUTF8String(jsOneString
, "1");
1476 assertEqualsAsUTF8String(jsCFString
, "A");
1477 assertEqualsAsUTF8String(jsCFStringWithCharacters
, "A");
1478 assertEqualsAsUTF8String(jsCFEmptyString
, "");
1479 assertEqualsAsUTF8String(jsCFEmptyStringWithCharacters
, "");
1481 checkConstnessInJSObjectNames();
1483 ASSERT(JSValueIsStrictEqual(context
, jsTrue
, jsTrue
));
1484 ASSERT(!JSValueIsStrictEqual(context
, jsOne
, jsOneString
));
1486 ASSERT(JSValueIsEqual(context
, jsOne
, jsOneString
, NULL
));
1487 ASSERT(!JSValueIsEqual(context
, jsTrue
, jsFalse
, NULL
));
1489 CFStringRef cfJSString
= JSStringCopyCFString(kCFAllocatorDefault
, jsCFIString
);
1490 CFStringRef cfJSEmptyString
= JSStringCopyCFString(kCFAllocatorDefault
, jsCFEmptyIString
);
1491 ASSERT(CFEqual(cfJSString
, cfString
));
1492 ASSERT(CFEqual(cfJSEmptyString
, cfEmptyString
));
1493 CFRelease(cfJSString
);
1494 CFRelease(cfJSEmptyString
);
1496 CFRelease(cfString
);
1497 CFRelease(cfEmptyString
);
1499 jsGlobalValue
= JSObjectMake(context
, NULL
, NULL
);
1500 makeGlobalNumberValue(context
);
1501 JSValueProtect(context
, jsGlobalValue
);
1502 JSGarbageCollect(context
);
1503 ASSERT(JSValueIsObject(context
, jsGlobalValue
));
1504 JSValueUnprotect(context
, jsGlobalValue
);
1505 JSValueUnprotect(context
, jsNumberValue
);
1507 JSStringRef goodSyntax
= JSStringCreateWithUTF8CString("x = 1;");
1508 const char* badSyntaxConstant
= "x := 1;";
1509 JSStringRef badSyntax
= JSStringCreateWithUTF8CString(badSyntaxConstant
);
1510 ASSERT(JSCheckScriptSyntax(context
, goodSyntax
, NULL
, 0, NULL
));
1511 ASSERT(!JSCheckScriptSyntax(context
, badSyntax
, NULL
, 0, NULL
));
1512 ASSERT(!JSScriptCreateFromString(contextGroup
, 0, 0, badSyntax
, 0, 0));
1513 ASSERT(!JSScriptCreateReferencingImmortalASCIIText(contextGroup
, 0, 0, badSyntaxConstant
, strlen(badSyntaxConstant
), 0, 0));
1520 result
= JSEvaluateScript(context
, goodSyntax
, NULL
, NULL
, 1, NULL
);
1522 ASSERT(JSValueIsEqual(context
, result
, jsOne
, NULL
));
1525 result
= JSEvaluateScript(context
, badSyntax
, NULL
, NULL
, 1, &exception
);
1527 ASSERT(JSValueIsObject(context
, exception
));
1529 JSStringRef array
= JSStringCreateWithUTF8CString("Array");
1530 JSObjectRef arrayConstructor
= JSValueToObject(context
, JSObjectGetProperty(context
, globalObject
, array
, NULL
), NULL
);
1531 JSStringRelease(array
);
1532 result
= JSObjectCallAsConstructor(context
, arrayConstructor
, 0, NULL
, NULL
);
1534 ASSERT(JSValueIsObject(context
, result
));
1535 ASSERT(JSValueIsInstanceOfConstructor(context
, result
, arrayConstructor
, NULL
));
1536 ASSERT(!JSValueIsInstanceOfConstructor(context
, JSValueMakeNull(context
), arrayConstructor
, NULL
));
1538 o
= JSValueToObject(context
, result
, NULL
);
1540 ASSERT(JSValueIsUndefined(context
, JSObjectGetPropertyAtIndex(context
, o
, 0, &exception
)));
1543 JSObjectSetPropertyAtIndex(context
, o
, 0, JSValueMakeNumber(context
, 1), &exception
);
1547 ASSERT(1 == JSValueToNumber(context
, JSObjectGetPropertyAtIndex(context
, o
, 0, &exception
), &exception
));
1550 JSStringRef functionBody
;
1551 JSObjectRef function
;
1554 functionBody
= JSStringCreateWithUTF8CString("rreturn Array;");
1555 JSStringRef line
= JSStringCreateWithUTF8CString("line");
1556 ASSERT(!JSObjectMakeFunction(context
, NULL
, 0, NULL
, functionBody
, NULL
, 1, &exception
));
1557 ASSERT(JSValueIsObject(context
, exception
));
1558 v
= JSObjectGetProperty(context
, JSValueToObject(context
, exception
, NULL
), line
, NULL
);
1559 assertEqualsAsNumber(v
, 1);
1560 JSStringRelease(functionBody
);
1561 JSStringRelease(line
);
1564 functionBody
= JSStringCreateWithUTF8CString("return Array;");
1565 function
= JSObjectMakeFunction(context
, NULL
, 0, NULL
, functionBody
, NULL
, 1, &exception
);
1566 JSStringRelease(functionBody
);
1568 ASSERT(JSObjectIsFunction(context
, function
));
1569 v
= JSObjectCallAsFunction(context
, function
, NULL
, 0, NULL
, NULL
);
1571 ASSERT(JSValueIsEqual(context
, v
, arrayConstructor
, NULL
));
1574 function
= JSObjectMakeFunction(context
, NULL
, 0, NULL
, jsEmptyIString
, NULL
, 0, &exception
);
1576 v
= JSObjectCallAsFunction(context
, function
, NULL
, 0, NULL
, &exception
);
1577 ASSERT(v
&& !exception
);
1578 ASSERT(JSValueIsUndefined(context
, v
));
1582 JSStringRef foo
= JSStringCreateWithUTF8CString("foo");
1583 JSStringRef argumentNames
[] = { foo
};
1584 functionBody
= JSStringCreateWithUTF8CString("return foo;");
1585 function
= JSObjectMakeFunction(context
, foo
, 1, argumentNames
, functionBody
, NULL
, 1, &exception
);
1586 ASSERT(function
&& !exception
);
1587 JSValueRef arguments
[] = { JSValueMakeNumber(context
, 2) };
1588 JSObjectCallAsFunction(context
, function
, NULL
, 1, arguments
, &exception
);
1589 JSStringRelease(foo
);
1590 JSStringRelease(functionBody
);
1592 string
= JSValueToStringCopy(context
, function
, NULL
);
1593 assertEqualsAsUTF8String(JSValueMakeString(context
, string
), "function foo(foo) { return foo;\n}");
1594 JSStringRelease(string
);
1596 JSStringRef print
= JSStringCreateWithUTF8CString("print");
1597 JSObjectRef printFunction
= JSObjectMakeFunctionWithCallback(context
, print
, print_callAsFunction
);
1598 JSObjectSetProperty(context
, globalObject
, print
, printFunction
, kJSPropertyAttributeNone
, NULL
);
1599 JSStringRelease(print
);
1601 ASSERT(!JSObjectSetPrivate(printFunction
, (void*)1));
1602 ASSERT(!JSObjectGetPrivate(printFunction
));
1604 JSStringRef myConstructorIString
= JSStringCreateWithUTF8CString("MyConstructor");
1605 JSObjectRef myConstructor
= JSObjectMakeConstructor(context
, NULL
, myConstructor_callAsConstructor
);
1606 JSObjectSetProperty(context
, globalObject
, myConstructorIString
, myConstructor
, kJSPropertyAttributeNone
, NULL
);
1607 JSStringRelease(myConstructorIString
);
1609 JSStringRef myBadConstructorIString
= JSStringCreateWithUTF8CString("MyBadConstructor");
1610 JSObjectRef myBadConstructor
= JSObjectMakeConstructor(context
, NULL
, myBadConstructor_callAsConstructor
);
1611 JSObjectSetProperty(context
, globalObject
, myBadConstructorIString
, myBadConstructor
, kJSPropertyAttributeNone
, NULL
);
1612 JSStringRelease(myBadConstructorIString
);
1614 ASSERT(!JSObjectSetPrivate(myConstructor
, (void*)1));
1615 ASSERT(!JSObjectGetPrivate(myConstructor
));
1617 string
= JSStringCreateWithUTF8CString("Base");
1618 JSObjectRef baseConstructor
= JSObjectMakeConstructor(context
, Base_class(context
), NULL
);
1619 JSObjectSetProperty(context
, globalObject
, string
, baseConstructor
, kJSPropertyAttributeNone
, NULL
);
1620 JSStringRelease(string
);
1622 string
= JSStringCreateWithUTF8CString("Derived");
1623 JSObjectRef derivedConstructor
= JSObjectMakeConstructor(context
, Derived_class(context
), NULL
);
1624 JSObjectSetProperty(context
, globalObject
, string
, derivedConstructor
, kJSPropertyAttributeNone
, NULL
);
1625 JSStringRelease(string
);
1627 string
= JSStringCreateWithUTF8CString("Derived2");
1628 JSObjectRef derived2Constructor
= JSObjectMakeConstructor(context
, Derived2_class(context
), NULL
);
1629 JSObjectSetProperty(context
, globalObject
, string
, derived2Constructor
, kJSPropertyAttributeNone
, NULL
);
1630 JSStringRelease(string
);
1632 o
= JSObjectMake(context
, NULL
, NULL
);
1633 JSObjectSetProperty(context
, o
, jsOneIString
, JSValueMakeNumber(context
, 1), kJSPropertyAttributeNone
, NULL
);
1634 JSObjectSetProperty(context
, o
, jsCFIString
, JSValueMakeNumber(context
, 1), kJSPropertyAttributeDontEnum
, NULL
);
1635 JSPropertyNameArrayRef nameArray
= JSObjectCopyPropertyNames(context
, o
);
1636 size_t expectedCount
= JSPropertyNameArrayGetCount(nameArray
);
1638 for (count
= 0; count
< expectedCount
; ++count
)
1639 JSPropertyNameArrayGetNameAtIndex(nameArray
, count
);
1640 JSPropertyNameArrayRelease(nameArray
);
1641 ASSERT(count
== 1); // jsCFString should not be enumerated
1643 JSValueRef argumentsArrayValues
[] = { JSValueMakeNumber(context
, 10), JSValueMakeNumber(context
, 20) };
1644 o
= JSObjectMakeArray(context
, sizeof(argumentsArrayValues
) / sizeof(JSValueRef
), argumentsArrayValues
, NULL
);
1645 string
= JSStringCreateWithUTF8CString("length");
1646 v
= JSObjectGetProperty(context
, o
, string
, NULL
);
1647 assertEqualsAsNumber(v
, 2);
1648 v
= JSObjectGetPropertyAtIndex(context
, o
, 0, NULL
);
1649 assertEqualsAsNumber(v
, 10);
1650 v
= JSObjectGetPropertyAtIndex(context
, o
, 1, NULL
);
1651 assertEqualsAsNumber(v
, 20);
1653 o
= JSObjectMakeArray(context
, 0, NULL
, NULL
);
1654 v
= JSObjectGetProperty(context
, o
, string
, NULL
);
1655 assertEqualsAsNumber(v
, 0);
1656 JSStringRelease(string
);
1658 JSValueRef argumentsDateValues
[] = { JSValueMakeNumber(context
, 0) };
1659 o
= JSObjectMakeDate(context
, 1, argumentsDateValues
, NULL
);
1660 if (timeZoneIsPST())
1661 assertEqualsAsUTF8String(o
, "Wed Dec 31 1969 16:00:00 GMT-0800 (PST)");
1663 string
= JSStringCreateWithUTF8CString("an error message");
1664 JSValueRef argumentsErrorValues
[] = { JSValueMakeString(context
, string
) };
1665 o
= JSObjectMakeError(context
, 1, argumentsErrorValues
, NULL
);
1666 assertEqualsAsUTF8String(o
, "Error: an error message");
1667 JSStringRelease(string
);
1669 string
= JSStringCreateWithUTF8CString("foo");
1670 JSStringRef string2
= JSStringCreateWithUTF8CString("gi");
1671 JSValueRef argumentsRegExpValues
[] = { JSValueMakeString(context
, string
), JSValueMakeString(context
, string2
) };
1672 o
= JSObjectMakeRegExp(context
, 2, argumentsRegExpValues
, NULL
);
1673 assertEqualsAsUTF8String(o
, "/foo/gi");
1674 JSStringRelease(string
);
1675 JSStringRelease(string2
);
1677 JSClassDefinition nullDefinition
= kJSClassDefinitionEmpty
;
1678 nullDefinition
.attributes
= kJSClassAttributeNoAutomaticPrototype
;
1679 JSClassRef nullClass
= JSClassCreate(&nullDefinition
);
1680 JSClassRelease(nullClass
);
1682 nullDefinition
= kJSClassDefinitionEmpty
;
1683 nullClass
= JSClassCreate(&nullDefinition
);
1684 JSClassRelease(nullClass
);
1686 functionBody
= JSStringCreateWithUTF8CString("return this;");
1687 function
= JSObjectMakeFunction(context
, NULL
, 0, NULL
, functionBody
, NULL
, 1, NULL
);
1688 JSStringRelease(functionBody
);
1689 v
= JSObjectCallAsFunction(context
, function
, NULL
, 0, NULL
, NULL
);
1690 ASSERT(JSValueIsEqual(context
, v
, globalObject
, NULL
));
1691 v
= JSObjectCallAsFunction(context
, function
, o
, 0, NULL
, NULL
);
1692 ASSERT(JSValueIsEqual(context
, v
, o
, NULL
));
1694 functionBody
= JSStringCreateWithUTF8CString("return eval(\"this\");");
1695 function
= JSObjectMakeFunction(context
, NULL
, 0, NULL
, functionBody
, NULL
, 1, NULL
);
1696 JSStringRelease(functionBody
);
1697 v
= JSObjectCallAsFunction(context
, function
, NULL
, 0, NULL
, NULL
);
1698 ASSERT(JSValueIsEqual(context
, v
, globalObject
, NULL
));
1699 v
= JSObjectCallAsFunction(context
, function
, o
, 0, NULL
, NULL
);
1700 ASSERT(JSValueIsEqual(context
, v
, o
, NULL
));
1702 const char* thisScript
= "this;";
1703 JSStringRef script
= JSStringCreateWithUTF8CString(thisScript
);
1704 v
= JSEvaluateScript(context
, script
, NULL
, NULL
, 1, NULL
);
1705 ASSERT(JSValueIsEqual(context
, v
, globalObject
, NULL
));
1706 v
= JSEvaluateScript(context
, script
, o
, NULL
, 1, NULL
);
1707 ASSERT(JSValueIsEqual(context
, v
, o
, NULL
));
1708 JSStringRelease(script
);
1710 JSScriptRef scriptObject
= JSScriptCreateReferencingImmortalASCIIText(contextGroup
, 0, 0, thisScript
, strlen(thisScript
), 0, 0);
1711 v
= JSScriptEvaluate(context
, scriptObject
, NULL
, NULL
);
1712 ASSERT(JSValueIsEqual(context
, v
, globalObject
, NULL
));
1713 v
= JSScriptEvaluate(context
, scriptObject
, o
, NULL
);
1714 ASSERT(JSValueIsEqual(context
, v
, o
, NULL
));
1715 JSScriptRelease(scriptObject
);
1717 script
= JSStringCreateWithUTF8CString("eval(this);");
1718 v
= JSEvaluateScript(context
, script
, NULL
, NULL
, 1, NULL
);
1719 ASSERT(JSValueIsEqual(context
, v
, globalObject
, NULL
));
1720 v
= JSEvaluateScript(context
, script
, o
, NULL
, 1, NULL
);
1721 ASSERT(JSValueIsEqual(context
, v
, o
, NULL
));
1722 JSStringRelease(script
);
1724 // Verify that creating a constructor for a class with no static functions does not trigger
1725 // an assert inside putDirect or lead to a crash during GC. <https://bugs.webkit.org/show_bug.cgi?id=25785>
1726 nullDefinition
= kJSClassDefinitionEmpty
;
1727 nullClass
= JSClassCreate(&nullDefinition
);
1728 JSObjectMakeConstructor(context
, nullClass
, 0);
1729 JSClassRelease(nullClass
);
1731 char* scriptUTF8
= createStringWithContentsOfFile(scriptPath
);
1733 printf("FAIL: Test script could not be loaded.\n");
1736 JSStringRef url
= JSStringCreateWithUTF8CString(scriptPath
);
1737 JSStringRef script
= JSStringCreateWithUTF8CString(scriptUTF8
);
1738 JSStringRef errorMessage
= 0;
1740 JSScriptRef scriptObject
= JSScriptCreateFromString(contextGroup
, url
, 1, script
, &errorMessage
, &errorLine
);
1741 ASSERT((!scriptObject
) != (!errorMessage
));
1742 if (!scriptObject
) {
1743 printf("FAIL: Test script did not parse\n\t%s:%d\n\t", scriptPath
, errorLine
);
1744 CFStringRef errorCF
= JSStringCopyCFString(kCFAllocatorDefault
, errorMessage
);
1747 JSStringRelease(errorMessage
);
1751 JSStringRelease(script
);
1752 result
= scriptObject
? JSScriptEvaluate(context
, scriptObject
, 0, &exception
) : 0;
1753 if (result
&& JSValueIsUndefined(context
, result
))
1754 printf("PASS: Test script executed successfully.\n");
1756 printf("FAIL: Test script returned unexpected value:\n");
1757 JSStringRef exceptionIString
= JSValueToStringCopy(context
, exception
, NULL
);
1758 CFStringRef exceptionCF
= JSStringCopyCFString(kCFAllocatorDefault
, exceptionIString
);
1759 CFShow(exceptionCF
);
1760 CFRelease(exceptionCF
);
1761 JSStringRelease(exceptionIString
);
1764 JSScriptRelease(scriptObject
);
1768 #if PLATFORM(MAC) || PLATFORM(IOS)
1769 JSStringRef currentCPUTimeStr
= JSStringCreateWithUTF8CString("currentCPUTime");
1770 JSObjectRef currentCPUTimeFunction
= JSObjectMakeFunctionWithCallback(context
, currentCPUTimeStr
, currentCPUTime_callAsFunction
);
1771 JSObjectSetProperty(context
, globalObject
, currentCPUTimeStr
, currentCPUTimeFunction
, kJSPropertyAttributeNone
, NULL
);
1772 JSStringRelease(currentCPUTimeStr
);
1774 /* Test script timeout: */
1775 JSContextGroupSetExecutionTimeLimit(contextGroup
, .10f
, shouldTerminateCallback
, 0);
1777 const char* loopForeverScript
= "var startTime = currentCPUTime(); while (true) { if (currentCPUTime() - startTime > .150) break; } ";
1778 JSStringRef script
= JSStringCreateWithUTF8CString(loopForeverScript
);
1782 shouldTerminateCallbackWasCalled
= false;
1783 startTime
= currentCPUTime();
1784 v
= JSEvaluateScript(context
, script
, NULL
, NULL
, 1, &exception
);
1785 endTime
= currentCPUTime();
1787 if (((endTime
- startTime
) < .150f
) && shouldTerminateCallbackWasCalled
)
1788 printf("PASS: script timed out as expected.\n");
1790 if (!((endTime
- startTime
) < .150f
))
1791 printf("FAIL: script did not timed out as expected.\n");
1792 if (!shouldTerminateCallbackWasCalled
)
1793 printf("FAIL: script timeout callback was not called.\n");
1798 printf("FAIL: TerminatedExecutionException was not thrown.\n");
1803 /* Test the script timeout's TerminatedExecutionException should NOT be catchable: */
1804 JSContextGroupSetExecutionTimeLimit(contextGroup
, 0.10f
, shouldTerminateCallback
, 0);
1806 const char* loopForeverScript
= "var startTime = currentCPUTime(); try { while (true) { if (currentCPUTime() - startTime > .150) break; } } catch(e) { }";
1807 JSStringRef script
= JSStringCreateWithUTF8CString(loopForeverScript
);
1811 shouldTerminateCallbackWasCalled
= false;
1812 startTime
= currentCPUTime();
1813 v
= JSEvaluateScript(context
, script
, NULL
, NULL
, 1, &exception
);
1814 endTime
= currentCPUTime();
1816 if (((endTime
- startTime
) >= .150f
) || !shouldTerminateCallbackWasCalled
) {
1817 if (!((endTime
- startTime
) < .150f
))
1818 printf("FAIL: script did not timed out as expected.\n");
1819 if (!shouldTerminateCallbackWasCalled
)
1820 printf("FAIL: script timeout callback was not called.\n");
1825 printf("PASS: TerminatedExecutionException was not catchable as expected.\n");
1827 printf("FAIL: TerminatedExecutionException was caught.\n");
1832 /* Test script timeout with no callback: */
1833 JSContextGroupSetExecutionTimeLimit(contextGroup
, .10f
, 0, 0);
1835 const char* loopForeverScript
= "var startTime = currentCPUTime(); while (true) { if (currentCPUTime() - startTime > .150) break; } ";
1836 JSStringRef script
= JSStringCreateWithUTF8CString(loopForeverScript
);
1840 startTime
= currentCPUTime();
1841 v
= JSEvaluateScript(context
, script
, NULL
, NULL
, 1, &exception
);
1842 endTime
= currentCPUTime();
1844 if (((endTime
- startTime
) < .150f
) && shouldTerminateCallbackWasCalled
)
1845 printf("PASS: script timed out as expected when no callback is specified.\n");
1847 if (!((endTime
- startTime
) < .150f
))
1848 printf("FAIL: script did not timed out as expected when no callback is specified.\n");
1853 printf("FAIL: TerminatedExecutionException was not thrown.\n");
1858 /* Test script timeout cancellation: */
1859 JSContextGroupSetExecutionTimeLimit(contextGroup
, 0.10f
, cancelTerminateCallback
, 0);
1861 const char* loopForeverScript
= "var startTime = currentCPUTime(); while (true) { if (currentCPUTime() - startTime > .150) break; } ";
1862 JSStringRef script
= JSStringCreateWithUTF8CString(loopForeverScript
);
1866 startTime
= currentCPUTime();
1867 v
= JSEvaluateScript(context
, script
, NULL
, NULL
, 1, &exception
);
1868 endTime
= currentCPUTime();
1870 if (((endTime
- startTime
) >= .150f
) && cancelTerminateCallbackWasCalled
&& !exception
)
1871 printf("PASS: script timeout was cancelled as expected.\n");
1873 if (((endTime
- startTime
) < .150) || exception
)
1874 printf("FAIL: script timeout was not cancelled.\n");
1875 if (!cancelTerminateCallbackWasCalled
)
1876 printf("FAIL: script timeout callback was not called.\n");
1881 printf("FAIL: Unexpected TerminatedExecutionException thrown.\n");
1886 /* Test script timeout extension: */
1887 JSContextGroupSetExecutionTimeLimit(contextGroup
, 0.100f
, extendTerminateCallback
, 0);
1889 const char* loopForeverScript
= "var startTime = currentCPUTime(); while (true) { if (currentCPUTime() - startTime > .500) break; } ";
1890 JSStringRef script
= JSStringCreateWithUTF8CString(loopForeverScript
);
1895 startTime
= currentCPUTime();
1896 v
= JSEvaluateScript(context
, script
, NULL
, NULL
, 1, &exception
);
1897 endTime
= currentCPUTime();
1898 deltaTime
= endTime
- startTime
;
1900 if ((deltaTime
>= .300f
) && (deltaTime
< .500f
) && (extendTerminateCallbackCalled
== 2) && exception
)
1901 printf("PASS: script timeout was extended as expected.\n");
1903 if (deltaTime
< .200f
)
1904 printf("FAIL: script timeout was not extended as expected.\n");
1905 else if (deltaTime
>= .500f
)
1906 printf("FAIL: script did not timeout.\n");
1908 if (extendTerminateCallbackCalled
< 1)
1909 printf("FAIL: script timeout callback was not called.\n");
1910 if (extendTerminateCallbackCalled
< 2)
1911 printf("FAIL: script timeout callback was not called after timeout extension.\n");
1914 printf("FAIL: TerminatedExecutionException was not thrown during timeout extension test.\n");
1919 #endif /* PLATFORM(MAC) || PLATFORM(IOS) */
1921 // Clear out local variables pointing at JSObjectRefs to allow their values to be collected
1925 globalObject
= NULL
;
1926 myConstructor
= NULL
;
1928 JSStringRelease(jsEmptyIString
);
1929 JSStringRelease(jsOneIString
);
1930 JSStringRelease(jsCFIString
);
1931 JSStringRelease(jsCFEmptyIString
);
1932 JSStringRelease(jsCFIStringWithCharacters
);
1933 JSStringRelease(jsCFEmptyIStringWithCharacters
);
1934 JSStringRelease(goodSyntax
);
1935 JSStringRelease(badSyntax
);
1937 JSGlobalContextRelease(context
);
1938 JSClassRelease(globalObjectClass
);
1940 // Test for an infinite prototype chain that used to be created. This test
1941 // passes if the call to JSObjectHasProperty() does not hang.
1943 JSClassDefinition prototypeLoopClassDefinition
= kJSClassDefinitionEmpty
;
1944 prototypeLoopClassDefinition
.staticFunctions
= globalObject_staticFunctions
;
1945 JSClassRef prototypeLoopClass
= JSClassCreate(&prototypeLoopClassDefinition
);
1946 JSGlobalContextRef prototypeLoopContext
= JSGlobalContextCreateInGroup(NULL
, prototypeLoopClass
);
1948 JSStringRef nameProperty
= JSStringCreateWithUTF8CString("name");
1949 JSObjectHasProperty(prototypeLoopContext
, JSContextGetGlobalObject(prototypeLoopContext
), nameProperty
);
1951 JSGlobalContextRelease(prototypeLoopContext
);
1952 JSClassRelease(prototypeLoopClass
);
1954 printf("PASS: Infinite prototype chain does not occur.\n");
1956 if (checkForCycleInPrototypeChain())
1957 printf("PASS: A cycle in a prototype chain can't be created.\n");
1959 printf("FAIL: A cycle in a prototype chain can be created.\n");
1964 printf("FAIL: Some tests failed.\n");
1968 printf("PASS: Program exited normally.\n");
1972 static char* createStringWithContentsOfFile(const char* fileName
)
1976 size_t buffer_size
= 0;
1977 size_t buffer_capacity
= 1024;
1978 buffer
= (char*)malloc(buffer_capacity
);
1980 FILE* f
= fopen(fileName
, "r");
1982 fprintf(stderr
, "Could not open file: %s\n", fileName
);
1986 while (!feof(f
) && !ferror(f
)) {
1987 buffer_size
+= fread(buffer
+ buffer_size
, 1, buffer_capacity
- buffer_size
, f
);
1988 if (buffer_size
== buffer_capacity
) { // guarantees space for trailing '\0'
1989 buffer_capacity
*= 2;
1990 buffer
= (char*)realloc(buffer
, buffer_capacity
);
1994 ASSERT(buffer_size
< buffer_capacity
);
1997 buffer
[buffer_size
] = '\0';