Oolite
Loading...
Searching...
No Matches
OOJavaScriptEngine.m
Go to the documentation of this file.
1/*
2
3OOJavaScriptEngine.m
4
5JavaScript support for Oolite
6Copyright (C) 2007-2013 David Taylor and Jens Ayton.
7
8This program is free software; you can redistribute it and/or
9modify it under the terms of the GNU General Public License
10as published by the Free Software Foundation; either version 2
11of the License, or (at your option) any later version.
12
13This program is distributed in the hope that it will be useful,
14but WITHOUT ANY WARRANTY; without even the implied warranty of
15MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
16GNU General Public License for more details.
17
18You should have received a copy of the GNU General Public License
19along with this program; if not, write to the Free Software
20Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston,
21MA 02110-1301, USA.
22
23*/
24
25#include <jsdbgapi.h>
28#import "OOJSScript.h"
29
31#import "Universe.h"
32#import "OOPlanetEntity.h"
34#import "OOWeakReference.h"
36#import "ResourceManager.h"
38#import "OOConstToJSString.h"
40#import "OOWaypointEntity.h"
41
42#import "OOJSGlobal.h"
44#import "OOJSMission.h"
45#import "OOJSVector.h"
46#import "OOJSQuaternion.h"
47#import "OOJSEntity.h"
48#import "OOJSShip.h"
49#import "OOJSStation.h"
50#import "OOJSDock.h"
51#import "OOJSVisualEffect.h"
52#import "OOJSExhaustPlume.h"
53#import "OOJSFlasher.h"
54#import "OOJSWormhole.h"
55#import "OOJSWaypoint.h"
56#import "OOJSPlayer.h"
57#import "OOJSPlayerShip.h"
58#import "OOJSManifest.h"
59#import "OOJSPlanet.h"
60#import "OOJSSystem.h"
61#import "OOJSOolite.h"
62#import "OOJSTimer.h"
63#import "OOJSClock.h"
64#import "OOJSSun.h"
65#import "OOJSWorldScripts.h"
66#import "OOJSSound.h"
67#import "OOJSSoundSource.h"
69#import "OOJSSystemInfo.h"
70#import "OOJSEquipmentInfo.h"
71#import "OOJSShipGroup.h"
73#import "OOJSFont.h"
74
76#import "OOLoggingExtended.h"
77
78#include <stdlib.h>
79
80
81#define OOJSENGINE_JSVERSION JSVERSION_ECMA_5
82#ifdef DEBUG
83#define JIT_OPTIONS 0
84#else
85#define JIT_OPTIONS JSOPTION_JIT | JSOPTION_METHODJIT | JSOPTION_PROFILING
86#endif
87#define OOJSENGINE_CONTEXT_OPTIONS JSOPTION_VAROBJFIX | JSOPTION_RELIMIT | JSOPTION_ANONFUNFIX | JIT_OPTIONS
88
89
90#define OOJS_STACK_SIZE 8192
91#define OOJS_RUNTIME_SIZE_MiB 256
92
93
95static unsigned sErrorHandlerStackSkip = 0;
96
97JSContext *gOOJSMainThreadContext = NULL;
98
99
100NSString * const kOOJavaScriptEngineWillResetNotification = @"org.aegidian.oolite OOJavaScriptEngine will reset";
101NSString * const kOOJavaScriptEngineDidResetNotification = @"org.aegidian.oolite OOJavaScriptEngine did reset";
102
103
104#if OOJSENGINE_MONITOR_SUPPORT
105
106@interface OOJavaScriptEngine (OOMonitorSupportInternal)
107
108- (void)sendMonitorError:(JSErrorReport *)errorReport
109 withMessage:(NSString *)message
110 inContext:(JSContext *)context;
111
112- (void)sendMonitorLogMessage:(NSString *)message
113 withMessageClass:(NSString *)messageClass
114 inContext:(JSContext *)context;
115
116@end
117
118#endif
119
120
121@interface OOJavaScriptEngine (Private)
122
123- (BOOL) lookUpStandardClassPointers;
124- (void) registerStandardObjectConverters;
125
126- (void) createMainThreadContext;
127- (void) destroyMainThreadContext;
128
129@end
130
131
132static void ReportJSError(JSContext *context, const char *message, JSErrorReport *report);
133
134static id JSArrayConverter(JSContext *context, JSObject *object);
135static id JSStringConverter(JSContext *context, JSObject *object);
136static id JSNumberConverter(JSContext *context, JSObject *object);
137static id JSBooleanConverter(JSContext *context, JSObject *object);
138
139
140static void UnregisterObjectConverters(void);
141static void UnregisterSubclasses(void);
142
143
144static void ReportJSError(JSContext *context, const char *message, JSErrorReport *report)
145{
146 NSString *severity = @"error";
147 NSString *messageText = nil;
148 NSString *lineBuf = nil;
149 NSString *messageClass = nil;
150 NSString *highlight = @"*****";
151 NSString *activeScript = nil;
153 BOOL showLocation = [jsEng showErrorLocations];
154
155 // Not OOJS_BEGIN_FULL_NATIVE() - we use JSAPI while paused.
157
158 jschar empty[1] = { 0 };
159 JSErrorReport blankReport =
160 {
161 .filename = "<unspecified file>",
162 .linebuf = "",
163 .uclinebuf = empty,
164 .uctokenptr = empty,
165 .ucmessage = empty
166 };
167 if (EXPECT_NOT(report == NULL)) report = &blankReport;
168 if (EXPECT_NOT(message == NULL || *message == '\0')) message = "<unspecified error>";
169
170 // Type of problem: error, warning or exception? (Strict flag wilfully ignored.)
171 if (report->flags & JSREPORT_EXCEPTION) severity = @"exception";
172 else if (report->flags & JSREPORT_WARNING)
173 {
174 severity = @"warning";
175 highlight = @"-----";
176 }
177
178 // The error message itself
179 messageText = [NSString stringWithUTF8String:message];
180
181 // Get offending line, if present, and trim trailing line breaks
182 lineBuf = [NSString stringWithUTF16String:report->uclinebuf];
183 while ([lineBuf hasSuffix:@"\n"] || [lineBuf hasSuffix:@"\r"]) lineBuf = [lineBuf substringToIndex:[lineBuf length] - 1];
184
185 // Get string for error number, for useful log message classes
186 NSDictionary *errorNames = [ResourceManager dictionaryFromFilesNamed:@"javascript-errors.plist" inFolder:@"Config" andMerge:YES];
187 NSString *errorNumberStr = [NSString stringWithFormat:@"%u", report->errorNumber];
188 NSString *errorName = [errorNames oo_stringForKey:errorNumberStr];
189 if (errorName == nil) errorName = errorNumberStr;
190
191 // Log message class
192 messageClass = [NSString stringWithFormat:@"script.javaScript.%@.%@", severity, errorName];
193
194 // Skip the rest if this is a warning being ignored.
195 if ((report->flags & JSREPORT_WARNING) == 0 || OOLogWillDisplayMessagesInClass(messageClass))
196 {
197 // First line: problem description
198 // avoid windows DEP exceptions!
200 activeScript = [[thisScript weakRefUnderlyingObject] displayName];
201 [thisScript release];
202
203 if (activeScript == nil) activeScript = @"<unidentified script>";
204 OOLog(messageClass, @"%@ JavaScript %@ (%@): %@", highlight, severity, activeScript, messageText);
205
206 if (showLocation && sErrorHandlerStackSkip == 0 && report->filename != NULL)
207 {
208 // Second line: where error occured, and line if provided. (The line is only provided for compile-time errors, not run-time errors.)
209 if ([lineBuf length] != 0)
210 {
211 OOLog(messageClass, @" %s, line %d: %@", report->filename, report->lineno, lineBuf);
212 }
213 else
214 {
215 OOLog(messageClass, @" %s, line %d.", report->filename, report->lineno);
216 }
217 }
218
219#ifndef NDEBUG
220 BOOL dump;
221 if (report->flags & JSREPORT_WARNING) dump = [jsEng dumpStackForWarnings];
222 else dump = [jsEng dumpStackForErrors];
223 if (dump) OOJSDumpStack(context);
224#endif
225
226#if OOJSENGINE_MONITOR_SUPPORT
227 JSExceptionState *exState = JS_SaveExceptionState(context);
228 [[OOJavaScriptEngine sharedEngine] sendMonitorError:report
229 withMessage:messageText
230 inContext:context];
231 JS_RestoreExceptionState(context, exState);
232#endif
233 }
234
236}
237
238
239//===========================================================================
240// JavaScript engine initialisation and shutdown
241//===========================================================================
242
243@implementation OOJavaScriptEngine
244
245+ (OOJavaScriptEngine *) sharedEngine
246{
247 if (sSharedEngine == nil) sSharedEngine = [[self alloc] init];
248
249 return sSharedEngine;
250}
251
252
253- (void) runMissionCallback
254{
256}
257
258
259- (id) init
260{
261 NSAssert(sSharedEngine == nil, @"Attempt to create multiple OOJavaScriptEngines.");
262
263 if (!(self = [super init])) return nil;
264 sSharedEngine = self;
265
266 JS_SetCStringsAreUTF8();
267
268 NSUserDefaults *defaults = [NSUserDefaults standardUserDefaults];
269#ifndef NDEBUG
270 /* Set stack trace preferences from preferences. These will be overriden
271 by the debug OXP script if installed, but being able to enable traces
272 without setting up the debug console could be useful for debugging
273 users' problems.
274 */
275 [self setDumpStackForErrors:[defaults boolForKey:@"dump-stack-for-errors"]];
276 [self setDumpStackForWarnings:[defaults boolForKey:@"dump-stack-for-warnings"]];
277#endif
278
279 assert(sizeof(jschar) == sizeof(unichar));
280
281 // initialize the JS run time, and return result in runtime.
282 uint32_t jsRuntimeInMiB = [defaults oo_intForKey:@"jsruntime-size-mib" defaultValue:OOJS_RUNTIME_SIZE_MiB];
283 _runtime = JS_NewRuntime(jsRuntimeInMiB * 1024L * 1024L);
284
285 // if runtime creation failed, end the program here.
286 if (_runtime == NULL)
287 {
288 OOLog(@"script.javaScript.init.error", @"***** FATAL ERROR: failed to create JavaScript runtime with size %uMiB.", jsRuntimeInMiB);
289 exit(1);
290 }
291
292 // OOJSTimeManagementInit() must be called before any context is created!
293 OOJSTimeManagementInit(self, _runtime);
294
295 [self createMainThreadContext];
296
297 return self;
298}
299
300
301- (void) createMainThreadContext
302{
303 NSAssert(gOOJSMainThreadContext == NULL, @"-[OOJavaScriptEngine createMainThreadContext] called while the main thread context exists.");
304
305 // create a context and associate it with the JS runtime.
306 gOOJSMainThreadContext = JS_NewContext(_runtime, OOJS_STACK_SIZE);
307
308 // if context creation failed, end the program here.
309 if (gOOJSMainThreadContext == NULL)
310 {
311 OOLog(@"script.javaScript.init.error", @"%@", @"***** FATAL ERROR: failed to create JavaScript context.");
312 exit(1);
313 }
314
316
319
320#if JS_GC_ZEAL
321 uint8_t gcZeal = [[NSUserDefaults standardUserDefaults] oo_unsignedCharForKey:@"js-gc-zeal"];
322 if (gcZeal > 0)
323 {
324 // Useful js-gc-zeal values are 0 (off), 1 and 2.
325 OOLog(@"script.javaScript.debug.gcZeal", @"Setting JavaScript garbage collector zeal to %u.", gcZeal);
326 JS_SetGCZeal(gOOJSMainThreadContext, gcZeal);
327 }
328#endif
329
330 JS_SetErrorReporter(gOOJSMainThreadContext, ReportJSError);
331
332 // Create the global object.
334
335 // Initialize the built-in JS objects and the global object.
336 JS_InitStandardClasses(gOOJSMainThreadContext, _globalObject);
337 if (![self lookUpStandardClassPointers])
338 {
339 OOLog(@"script.javaScript.init.error", @"%@", @"***** FATAL ERROR: failed to look up standard JavaScript classes.");
340 exit(1);
341 }
342 [self registerStandardObjectConverters];
343
346
347 // Initialize Oolite classes.
355 InitOOJSShip(gOOJSMainThreadContext, _globalObject);
357 InitOOJSDock(gOOJSMainThreadContext, _globalObject);
366 InitOOJSSun(gOOJSMainThreadContext, _globalObject);
379 InitOOJSFont(gOOJSMainThreadContext, _globalObject);
380
381 // Run prefix scripts.
382 [OOJSScript jsScriptFromFileNamed:@"oolite-global-prefix.js"
383 properties:[NSDictionary dictionaryWithObject:JSSpecialFunctionsObjectWrapper(gOOJSMainThreadContext)
384 forKey:@"special"]];
385
387
388 OOLog(@"script.javaScript.init.success", @"%@", @"Set up JavaScript context.");
389}
390
391
392- (void) destroyMainThreadContext
393{
394 if (gOOJSMainThreadContext != NULL)
395 {
396 JSContext *context = OOJSAcquireContext();
397 JS_ClearScope(gOOJSMainThreadContext, _globalObject);
398
399 _globalObject = NULL;
400 _objectClass = NULL;
401 _stringClass = NULL;
402 _arrayClass = NULL;
403 _numberClass = NULL;
404 _booleanClass = NULL;
405
409
410 OOJSRelinquishContext(context);
411
412 _globalObject = NULL;
413 JS_DestroyContext(gOOJSMainThreadContext); // Forces unconditional GC.
415 }
416}
417
418
419- (BOOL) reset
420{
421 NSAssert(gOOJSMainThreadContext != NULL, @"JavaScript engine not active. Can't reset.");
422
424
425# if 0
426 // deferred JS reset - test harness.
427 static int counter = 3; // loading a savegame with different strict mode calls js reset twice
428 if (counter-- == 0) {
429 counter = 3;
430 OOLog(@"script.javascript.init.error", @"%@", @"JavaScript processes still pending. Can't reset JavaScript engine.");
431 return NO;
432 }
433 else
434 {
435 OOLog(@"script.javascript.init", @"%@", @"JavaScript reset successful.");
436 }
437#endif
438
439#if JS_THREADSAFE
440 //NSAssert(!JS_IsInRequest(gOOJSMainThreadContext), @"JavaScript processes still pending. Can't reset JavaScript engine.");
441
443 {
444 // some threads are still pending, this should mean timers are still being removed.
445 OOLog(@"script.javascript.init.error", @"%@", @"JavaScript processes still pending. Can't reset JavaScript engine.");
446 return NO;
447 }
448 else
449 {
450 OOLog(@"script.javascript.init", @"%@", @"JavaScript reset successful.");
451 }
452#endif
453
454 JSContext *context = OOJSAcquireContext();
455 [[NSNotificationCenter defaultCenter] postNotificationName:kOOJavaScriptEngineWillResetNotification object:self];
456 OOJSRelinquishContext(context);
457
458 [self destroyMainThreadContext];
459 [self createMainThreadContext];
460
461 context = OOJSAcquireContext();
462 [[NSNotificationCenter defaultCenter] postNotificationName:kOOJavaScriptEngineDidResetNotification object:self];
463 OOJSRelinquishContext(context);
464
465 [self garbageCollectionOpportunity:YES];
466 return YES;
467}
468
469
470- (void) dealloc
471{
473
475
476 [self destroyMainThreadContext];
477 JS_DestroyRuntime(_runtime);
478
479 [super dealloc];
480}
481
482
483- (JSObject *) globalObject
484{
485 return _globalObject;
486}
487
488
489- (BOOL) callJSFunction:(jsval)function
490 forObject:(JSObject *)jsThis
491 argc:(uintN)argc
492 argv:(jsval *)argv
493 result:(jsval *)outResult
494{
495 JSContext *context = NULL;
496 BOOL result;
497
498 NSParameterAssert(OOJSValueIsFunction(context, function));
499
500 context = OOJSAcquireContext();
501
503 result = JS_CallFunctionValue(context, jsThis, function, argc, argv, outResult);
505
506 JS_ReportPendingException(context);
507 OOJSRelinquishContext(context);
508
509 return result;
510}
511
512
513- (void) removeGCObjectRoot:(JSObject **)rootPtr
514{
515 JSContext *context = OOJSAcquireContext();
516 JS_RemoveObjectRoot(context, rootPtr);
517 OOJSRelinquishContext(context);
518}
519
520
521- (void) removeGCValueRoot:(jsval *)rootPtr
522{
523 JSContext *context = OOJSAcquireContext();
524 JS_RemoveValueRoot(context, rootPtr);
525 OOJSRelinquishContext(context);
526}
527
528
529- (void) garbageCollectionOpportunity:(BOOL)force
530{
531 JSContext *context = OOJSAcquireContext();
532 if (force)
533 {
534 JS_GC(context);
535 }
536 else
537 {
538 JS_MaybeGC(context);
539 }
540 OOJSRelinquishContext(context);
541}
542
543
544- (BOOL) showErrorLocations
545{
546 return _showErrorLocations;
547}
548
549
550- (void) setShowErrorLocations:(BOOL)value
551{
552 _showErrorLocations = !!value;
553}
554
555
556- (JSClass *) objectClass
557{
558 return _objectClass;
559}
560
561
562- (JSClass *) stringClass
563{
564 return _stringClass;
565}
566
567
568- (JSClass *) arrayClass
569{
570 return _arrayClass;
571}
572
573
574- (JSClass *) numberClass
575{
576 return _numberClass;
577}
578
579
580- (JSClass *) booleanClass
581{
582 return _booleanClass;
583}
584
585
586- (BOOL) lookUpStandardClassPointers
587{
588 JSObject *templateObject = NULL;
589
590 templateObject = JS_NewObject(gOOJSMainThreadContext, NULL, NULL, NULL);
591 if (EXPECT_NOT(templateObject == NULL)) return NO;
592 _objectClass = OOJSGetClass(gOOJSMainThreadContext, templateObject);
593
594 if (EXPECT_NOT(!JS_ValueToObject(gOOJSMainThreadContext, JS_GetEmptyStringValue(gOOJSMainThreadContext), &templateObject))) return NO;
595 _stringClass = OOJSGetClass(gOOJSMainThreadContext, templateObject);
596
597 templateObject = JS_NewArrayObject(gOOJSMainThreadContext, 0, NULL);
598 if (EXPECT_NOT(templateObject == NULL)) return NO;
599 _arrayClass = OOJSGetClass(gOOJSMainThreadContext, templateObject);
600
601 if (EXPECT_NOT(!JS_ValueToObject(gOOJSMainThreadContext, INT_TO_JSVAL(0), &templateObject))) return NO;
602 _numberClass = OOJSGetClass(gOOJSMainThreadContext, templateObject);
603
604 if (EXPECT_NOT(!JS_ValueToObject(gOOJSMainThreadContext, JSVAL_FALSE, &templateObject))) return NO;
605 _booleanClass = OOJSGetClass(gOOJSMainThreadContext, templateObject);
606
607 return YES;
608}
609
610
611- (void) registerStandardObjectConverters
612{
618}
619
620
621#ifndef NDEBUG
622static JSTrapStatus DebuggerHook(JSContext *context, JSScript *script, jsbytecode *pc, jsval *rval, void *closure)
623{
625
626 OOLog(@"script.javaScript.debugger", @"debugger invoked during %@:", [[OOJSScript currentlyRunningScript] displayName]);
627 OOJSDumpStack(context);
628
630
631 return JSTRAP_CONTINUE;
632}
633
634
635- (BOOL) dumpStackForErrors
636{
637 return _dumpStackForErrors;
638}
639
640
641- (void) setDumpStackForErrors:(BOOL)value
642{
643 _dumpStackForErrors = !!value;
644}
645
646
647- (BOOL) dumpStackForWarnings
648{
649 return _dumpStackForWarnings;
650}
651
652
653- (void) setDumpStackForWarnings:(BOOL)value
654{
655 _dumpStackForWarnings = !!value;
656}
657
658
659- (void) enableDebuggerStatement
660{
661 JS_SetDebuggerHandler(_runtime, DebuggerHook, self);
662}
663#endif
664
665@end
666
667
668#if OOJSENGINE_MONITOR_SUPPORT
669
670@implementation OOJavaScriptEngine (OOMonitorSupport)
671
672- (void) setMonitor:(id<OOJavaScriptEngineMonitor>)inMonitor
673{
674 [_monitor autorelease];
675 _monitor = [inMonitor retain];
676}
677
678@end
679
680
681@implementation OOJavaScriptEngine (OOMonitorSupportInternal)
682
683- (void) sendMonitorError:(JSErrorReport *)errorReport
684 withMessage:(NSString *)message
685 inContext:(JSContext *)theContext
686{
687 if ([_monitor respondsToSelector:@selector(jsEngine:context:error:stackSkip:showingLocation:withMessage:)])
688 {
689 [_monitor jsEngine:self context:theContext error:errorReport stackSkip:sErrorHandlerStackSkip showingLocation:[self showErrorLocations] withMessage:message];
690 }
691}
692
693
694- (void) sendMonitorLogMessage:(NSString *)message
695 withMessageClass:(NSString *)messageClass
696 inContext:(JSContext *)theContext
697{
698 if ([_monitor respondsToSelector:@selector(jsEngine:context:logMessage:ofClass:)])
699 {
700 [_monitor jsEngine:self context:theContext logMessage:message ofClass:messageClass];
701 }
702}
703
704@end
705
706#endif
707
708
709#ifndef NDEBUG
710
711static void DumpVariable(JSContext *context, JSPropertyDesc *prop)
712{
713 NSString *name = OOStringFromJSValueEvenIfNull(context, prop->id);
714 NSString *value = OOJSDescribeValue(context, prop->value, YES);
715
716 enum
717 {
718 kInterestingFlags = ~(JSPD_ENUMERATE | JSPD_PERMANENT | JSPD_VARIABLE | JSPD_ARGUMENT)
719 };
720
721 NSString *flagStr = @"";
722 if ((prop->flags & kInterestingFlags) != 0)
723 {
724 NSMutableArray *flags = [NSMutableArray array];
725 if (prop->flags & JSPD_READONLY) [flags addObject:@"read-only"];
726 if (prop->flags & JSPD_ALIAS) [flags addObject:[NSString stringWithFormat:@"alias (%@)", OOJSDescribeValue(context, prop->alias, YES)]];
727 if (prop->flags & JSPD_EXCEPTION) [flags addObject:@"exception"];
728 if (prop->flags & JSPD_ERROR) [flags addObject:@"error"];
729
730 flagStr = [NSString stringWithFormat:@" [%@]", [flags componentsJoinedByString:@", "]];
731 }
732
733 OOLog(@"script.javaScript.stackTrace", @" %@: %@%@", name, value, flagStr);
734}
735
736
737void OOJSDumpStack(JSContext *context)
738{
739 NSAutoreleasePool *pool = [[NSAutoreleasePool alloc] init];
740
741 @try
742 {
743 JSStackFrame *frame = NULL;
744 unsigned idx = 0;
745 unsigned skip = sErrorHandlerStackSkip;
746
747 while (JS_FrameIterator(context, &frame) != NULL)
748 {
749 JSScript *script = JS_GetFrameScript(context, frame);
750 NSString *desc = nil;
751 JSPropertyDescArray properties = { 0 , NULL };
752 BOOL gotProperties = NO;
753
754 idx++;
755
756 if (!JS_IsScriptFrame(context, frame))
757 {
758 continue;
759 }
760
761 if (skip != 0)
762 {
763 skip--;
764 continue;
765 }
766
767 if (script != NULL)
768 {
769 NSString *location = OOJSDescribeLocation(context, frame);
770 JSObject *scope = JS_GetFrameScopeChain(context, frame);
771
772 if (scope != NULL) gotProperties = JS_GetPropertyDescArray(context, scope, &properties);
773
774 NSString *funcDesc = nil;
775 JSFunction *function = JS_GetFrameFunction(context, frame);
776 if (function != NULL)
777 {
778 JSString *funcName = JS_GetFunctionId(function);
779 if (funcName != NULL)
780 {
781 funcDesc = OOStringFromJSString(context, funcName);
782 if (!JS_IsConstructorFrame(context, frame))
783 {
784 funcDesc = [funcDesc stringByAppendingString:@"()"];
785 }
786 else
787 {
788 funcDesc = [NSString stringWithFormat:@"new %@()", funcDesc];
789 }
790
791 }
792 else
793 {
794 funcDesc = @"<anonymous function>";
795 }
796 }
797 else
798 {
799 funcDesc = @"<not a function frame>";
800 }
801
802 desc = [NSString stringWithFormat:@"(%@) %@", location, funcDesc];
803 }
804 else if (JS_IsDebuggerFrame(context, frame))
805 {
806 desc = @"<debugger frame>";
807 }
808 else
809 {
810 desc = @"<Oolite native>";
811 }
812
813 OOLog(@"script.javaScript.stackTrace", @"%2u %@", idx - 1, desc);
814
815 if (gotProperties)
816 {
817 jsval this;
818 if (JS_GetFrameThis(context, frame, &this))
819 {
820 static BOOL haveThis = NO;
821 static jsval thisAtom;
822 if (EXPECT_NOT(!haveThis))
823 {
824 thisAtom = STRING_TO_JSVAL(JS_InternString(context, "this"));
825 haveThis = YES;
826 }
827 JSPropertyDesc thisDesc = { .id = thisAtom, .value = this };
828 DumpVariable(context, &thisDesc);
829 }
830
831 // Dump arguments.
832 unsigned i;
833 for (i = 0; i < properties.length; i++)
834 {
835 JSPropertyDesc *prop = &properties.array[i];
836 if (prop->flags & JSPD_ARGUMENT) DumpVariable(context, prop);
837 }
838
839 // Dump locals.
840 for (i = 0; i < properties.length; i++)
841 {
842 JSPropertyDesc *prop = &properties.array[i];
843 if (prop->flags & JSPD_VARIABLE) DumpVariable(context, prop);
844 }
845
846 // Dump anything else.
847 for (i = 0; i < properties.length; i++)
848 {
849 JSPropertyDesc *prop = &properties.array[i];
850 if (!(prop->flags & (JSPD_ARGUMENT | JSPD_VARIABLE))) DumpVariable(context, prop);
851 }
852
853 JS_PutPropertyDescArray(context, &properties);
854 }
855 }
856 }
857 @catch (NSException *exception)
858 {
859 OOLog(kOOLogException, @"Exception during JavaScript stack trace: %@:%@", [exception name], [exception reason]);
860 }
861
862 [pool release];
863}
864
865
866static const char *sConsoleScriptName; // Lifetime is lifetime of script object, which is forever.
867static NSUInteger sConsoleEvalLineNo;
868
869
870static void GetLocationNameAndLine(JSContext *context, JSStackFrame *stackFrame, const char **name, NSUInteger *line)
871{
872 NSCParameterAssert(context != NULL && stackFrame != NULL && name != NULL && line != NULL);
873
874 *name = NULL;
875 *line = 0;
876
877 JSScript *script = JS_GetFrameScript(context, stackFrame);
878 if (script != NULL)
879 {
880 *name = JS_GetScriptFilename(context, script);
881 if (name != NULL)
882 {
883 jsbytecode *PC = JS_GetFramePC(context, stackFrame);
884 *line = JS_PCToLineNumber(context, script, PC);
885 }
886 }
887 else if (JS_IsDebuggerFrame(context, stackFrame))
888 {
889 *name = "<debugger frame>";
890 }
891}
892
893
894NSString *OOJSDescribeLocation(JSContext *context, JSStackFrame *stackFrame)
895{
896 NSCParameterAssert(context != NULL && stackFrame != NULL);
897
898 const char *fileName;
899 NSUInteger lineNo;
900 GetLocationNameAndLine(context, stackFrame, &fileName, &lineNo);
901 if (fileName == NULL) return nil;
902
903 // If this stops working, we probably need to switch to strcmp().
904 if (fileName == sConsoleScriptName && lineNo >= sConsoleEvalLineNo) return @"<console input>";
905
906 // Objectify it.
907 NSString *fileNameObj = [NSString stringWithUTF8String:fileName];
908 if (fileNameObj == nil) fileNameObj = [NSString stringWithCString:fileName encoding:NSISOLatin1StringEncoding];
909 if (fileNameObj == nil) return nil;
910
911 NSString *shortFileName = [fileNameObj lastPathComponent];
912 if (![[shortFileName lowercaseString] isEqualToString:@"script.js"]) fileNameObj = shortFileName;
913
914 return [NSString stringWithFormat:@"%@:%llu", fileNameObj, lineNo];
915}
916
917
918void OOJSMarkConsoleEvalLocation(JSContext *context, JSStackFrame *stackFrame)
919{
921}
922#endif
923
924
925void OOJSInitJSIDCachePRIVATE(const char *name, jsid *idCache)
926{
927 NSCParameterAssert(name != NULL && name[0] != '\0' && idCache != NULL);
928
929 JSContext *context = OOJSAcquireContext();
930
931 JSString *string = JS_InternString(context, name);
932 if (EXPECT_NOT(string == NULL))
933 {
934 [NSException raise:NSGenericException format:@"Failed to initialize JS ID cache for \"%s\".", name];
935 }
936
937 *idCache = INTERNED_STRING_TO_JSID(string);
938
939 OOJSRelinquishContext(context);
940}
941
942
943jsid OOJSIDFromString(NSString *string)
944{
945 if (EXPECT_NOT(string == nil)) return JSID_VOID;
946
947 JSContext *context = OOJSAcquireContext();
948
949 enum { kStackBufSize = 1024 };
950 unichar stackBuf[kStackBufSize];
951 unichar *buffer;
952 size_t length = [string length];
953 if (length < kStackBufSize)
954 {
955 buffer = stackBuf;
956 }
957 else
958 {
959 buffer = malloc(sizeof (unichar) * length);
960 if (EXPECT_NOT(buffer == NULL)) return JSID_VOID;
961 }
962 [string getCharacters:buffer];
963
964 JSString *jsString = JS_InternUCStringN(context, buffer, length);
965
966 if (EXPECT_NOT(buffer != stackBuf)) free(buffer);
967
968 OOJSRelinquishContext(context);
969
970 if (EXPECT(jsString != NULL)) return INTERNED_STRING_TO_JSID(jsString);
971 else return JSID_VOID;
972}
973
974
975NSString *OOStringFromJSID(jsid propID)
976{
977 JSContext *context = OOJSAcquireContext();
978
979 jsval value;
980 NSString *result = nil;
981 if (JS_IdToValue(context, propID, &value))
982 {
983 result = OOStringFromJSString(context, JS_ValueToString(context, value));
984 }
985
986 OOJSRelinquishContext(context);
987
988 return result;
989}
990
991
992static NSString *CallerPrefix(NSString *scriptClass, NSString *function)
993{
994 if (function == nil) return @"";
995 if (scriptClass == nil) return [function stringByAppendingString:@": "];
996 return [NSString stringWithFormat:@"%@.%@: ", scriptClass, function];
997}
998
999
1000void OOJSReportError(JSContext *context, NSString *format, ...)
1001{
1002 va_list args;
1003
1004 va_start(args, format);
1005 OOJSReportErrorWithArguments(context, format, args);
1006 va_end(args);
1007}
1008
1009
1010void OOJSReportErrorForCaller(JSContext *context, NSString *scriptClass, NSString *function, NSString *format, ...)
1011{
1012 va_list args;
1013 NSString *msg = nil;
1014
1015 @try
1016 {
1017 va_start(args, format);
1018 msg = [[NSString alloc] initWithFormat:format arguments:args];
1019 va_end(args);
1020
1021 OOJSReportError(context, @"%@%@", CallerPrefix(scriptClass, function), msg);
1022 }
1023 @catch (id exception)
1024 {
1025 // Squash any secondary errors during error handling.
1026 }
1027 [msg release];
1028}
1029
1030
1031void OOJSReportErrorWithArguments(JSContext *context, NSString *format, va_list args)
1032{
1033 NSString *msg = nil;
1034
1035 NSCParameterAssert(JS_IsInRequest(context));
1036
1037 @try
1038 {
1039 msg = [[NSString alloc] initWithFormat:format arguments:args];
1040 JS_ReportError(context, "%s", [msg UTF8String]);
1041 }
1042 @catch (id exception)
1043 {
1044 // Squash any secondary errors during error handling.
1045 }
1046 [msg release];
1047}
1048
1049
1050void OOJSReportWrappedException(JSContext *context, id exception)
1051{
1052 if (!JS_IsExceptionPending(context))
1053 {
1054 if ([exception isKindOfClass:[NSException class]]) OOJSReportError(context, @"Native exception: %@", [exception reason]);
1055 else OOJSReportError(context, @"Unidentified native exception");
1056 }
1057 // Else, let the pending exception propagate.
1058}
1059
1060
1061#ifndef NDEBUG
1062
1063void OOJSUnreachable(const char *function, const char *file, unsigned line)
1064{
1065 OOLog(@"fatal.unreachable", @"Supposedly unreachable statement reached in %s (%@:%u) -- terminating.", function, OOLogAbbreviatedFileName(file), line);
1066 abort();
1067}
1068
1069#endif
1070
1071
1072void OOJSReportWarning(JSContext *context, NSString *format, ...)
1073{
1074 va_list args;
1075
1076 va_start(args, format);
1077 OOJSReportWarningWithArguments(context, format, args);
1078 va_end(args);
1079}
1080
1081
1082void OOJSReportWarningForCaller(JSContext *context, NSString *scriptClass, NSString *function, NSString *format, ...)
1083{
1084 va_list args;
1085 NSString *msg = nil;
1086
1087 @try
1088 {
1089 va_start(args, format);
1090 msg = [[NSString alloc] initWithFormat:format arguments:args];
1091 va_end(args);
1092
1093 OOJSReportWarning(context, @"%@%@", CallerPrefix(scriptClass, function), msg);
1094 }
1095 @catch (id exception)
1096 {
1097 // Squash any secondary errors during error handling.
1098 }
1099 [msg release];
1100}
1101
1102
1103void OOJSReportWarningWithArguments(JSContext *context, NSString *format, va_list args)
1104{
1105 NSString *msg = nil;
1106
1107 @try
1108 {
1109 msg = [[NSString alloc] initWithFormat:format arguments:args];
1110 JS_ReportWarning(context, "%s", [msg UTF8String]);
1111 }
1112 @catch (id exception)
1113 {
1114 // Squash any secondary errors during error handling.
1115 }
1116 [msg release];
1117}
1118
1119
1120void OOJSReportBadPropertySelector(JSContext *context, JSObject *thisObj, jsid propID, JSPropertySpec *propertySpec)
1121{
1122 NSString *propName = OOStringFromJSPropertyIDAndSpec(context, propID, propertySpec);
1123 const char *className = OOJSGetClass(context, thisObj)->name;
1124
1125 OOJSReportError(context, @"Invalid property identifier %@ for instance of %s.", propName, className);
1126}
1127
1128
1129void OOJSReportBadPropertyValue(JSContext *context, JSObject *thisObj, jsid propID, JSPropertySpec *propertySpec, jsval value)
1130{
1131 NSString *propName = OOStringFromJSPropertyIDAndSpec(context, propID, propertySpec);
1132 const char *className = OOJSGetClass(context, thisObj)->name;
1133 NSString *valueDesc = OOJSDescribeValue(context, value, YES);
1134
1135 OOJSReportError(context, @"Cannot set property %@ of instance of %s to invalid value %@.", propName, className, valueDesc);
1136}
1137
1138
1139void OOJSReportBadArguments(JSContext *context, NSString *scriptClass, NSString *function, uintN argc, jsval *argv, NSString *message, NSString *expectedArgsDescription)
1140{
1141 @try
1142 {
1143 if (message == nil) message = @"Invalid arguments";
1144 message = [NSString stringWithFormat:@"%@ %@", message, [NSString stringWithJavaScriptParameters:argv count:argc inContext:context]];
1145 if (expectedArgsDescription != nil) message = [NSString stringWithFormat:@"%@ -- expected %@", message, expectedArgsDescription];
1146
1147 OOJSReportErrorForCaller(context, scriptClass, function, @"%@.", message);
1148 }
1149 @catch (id exception)
1150 {
1151 // Squash any secondary errors during error handling.
1152 }
1153}
1154
1155
1157{
1159}
1160
1161
1162BOOL OOJSArgumentListGetNumber(JSContext *context, NSString *scriptClass, NSString *function, uintN argc, jsval *argv, double *outNumber, uintN *outConsumed)
1163{
1164 if (OOJSArgumentListGetNumberNoError(context, argc, argv, outNumber, outConsumed))
1165 {
1166 return YES;
1167 }
1168 else
1169 {
1170 OOJSReportBadArguments(context, scriptClass, function, argc, argv,
1171 @"Expected number, got", NULL);
1172 return NO;
1173 }
1174}
1175
1176
1177BOOL OOJSArgumentListGetNumberNoError(JSContext *context, uintN argc, jsval *argv, double *outNumber, uintN *outConsumed)
1178{
1180
1181 double value;
1182
1183 NSCParameterAssert(context != NULL && (argv != NULL || argc == 0) && outNumber != NULL);
1184
1185 // Get value, if possible.
1186 if (EXPECT_NOT(!JS_ValueToNumber(context, argv[0], &value) || isnan(value)))
1187 {
1188 if (outConsumed != NULL) *outConsumed = 0;
1189 return NO;
1190 }
1191
1192 // Success.
1193 *outNumber = value;
1194 if (outConsumed != NULL) *outConsumed = 1;
1195 return YES;
1196
1198}
1199
1200
1201static JSObject *JSArrayFromNSArray(JSContext *context, NSArray *array)
1202{
1204
1205 JSObject *result = NULL;
1206
1207 if (array == nil) return NULL;
1208
1209 @try
1210 {
1211 NSUInteger fullCount = [array count];
1212 if (EXPECT_NOT(fullCount > INT32_MAX))
1213 {
1214 return NULL;
1215 }
1216
1217 uint32_t i, count = (int32_t)fullCount;
1218
1219 result = JS_NewArrayObject(context, 0, NULL);
1220 if (result != NULL)
1221 {
1222 for (i = 0; i != count; ++i)
1223 {
1224 jsval value = [[array objectAtIndex:i] oo_jsValueInContext:context];
1225 BOOL OK = JS_SetElement(context, result, i, &value);
1226
1227 if (EXPECT_NOT(!OK))
1228 {
1229 result = NULL;
1230 break;
1231 }
1232 }
1233 }
1234 }
1235 @catch (id ex)
1236 {
1237 result = NULL;
1238 }
1239
1240 return (JSObject *)result;
1241
1243}
1244
1245
1246static BOOL JSNewNSArrayValue(JSContext *context, NSArray *array, jsval *value)
1247{
1249
1250 JSObject *object = NULL;
1251 BOOL OK = YES;
1252
1253 if (value == NULL) return NO;
1254
1255 // NOTE: should be called within a local root scope or have *value be a set root for GC reasons.
1256 if (!JS_EnterLocalRootScope(context)) return NO;
1257
1258 object = JSArrayFromNSArray(context, array);
1259 if (object == NULL)
1260 {
1261 *value = JSVAL_VOID;
1262 OK = NO;
1263 }
1264 else
1265 {
1266 *value = OBJECT_TO_JSVAL(object);
1267 }
1268
1269 JS_LeaveLocalRootScopeWithResult(context, *value);
1270 return OK;
1271
1273}
1274
1275
1276/* Convert an NSDictionary to a JavaScript Object.
1277 Only properties whose keys are either strings or non-negative NSNumbers,
1278 and whose values have a non-void JS representation, are converted.
1279*/
1280static JSObject *JSObjectFromNSDictionary(JSContext *context, NSDictionary *dictionary)
1281{
1283
1284 JSObject *result = NULL;
1285 BOOL OK = YES;
1286 id key = nil;
1287 jsval value;
1288 jsint index;
1289
1290 if (dictionary == nil) return NULL;
1291
1292 @try
1293 {
1294 result = JS_NewObject(context, NULL, NULL, NULL); // create object of class Object
1295 if (result != NULL)
1296 {
1297 foreachkey (key, dictionary)
1298 {
1299 if ([key isKindOfClass:[NSString class]] && [key length] != 0)
1300 {
1301#ifndef __GNUC__
1302 value = [[dictionary objectForKey:key] oo_jsValueInContext:context];
1303#else
1304#if __GNUC__ > 4 || __GNUC_MINOR__ > 6
1305 value = [[dictionary objectForKey:key] oo_jsValueInContext:context];
1306#else
1307 // GCC before 4.7 seems to have problems with this
1308 // bit if the object is a weakref, causing crashes
1309 // in docking code.
1310 id tmp = [dictionary objectForKey:key];
1311 if ([tmp respondsToSelector:@selector(weakRefUnderlyingObject)])
1312 {
1313 tmp = [tmp weakRefUnderlyingObject];
1314 }
1315 value = [tmp oo_jsValueInContext:context];
1316#endif
1317#endif
1318 if (!JSVAL_IS_VOID(value))
1319 {
1320 OK = JS_SetPropertyById(context, result, OOJSIDFromString(key), &value);
1321 if (EXPECT_NOT(!OK)) break;
1322 }
1323 }
1324 else if ([key isKindOfClass:[NSNumber class]])
1325 {
1326 index = [key intValue];
1327 if (0 < index)
1328 {
1329 value = [[dictionary objectForKey:key] oo_jsValueInContext:context];
1330 if (!JSVAL_IS_VOID(value))
1331 {
1332 OK = JS_SetElement(context, (JSObject *)result, index, &value);
1333 if (EXPECT_NOT(!OK)) break;
1334 }
1335 }
1336 }
1337
1338 if (EXPECT_NOT(!OK)) break;
1339 }
1340 }
1341 }
1342 @catch (id exception)
1343 {
1344 OK = NO;
1345 }
1346
1347 if (EXPECT_NOT(!OK))
1348 {
1349 result = NULL;
1350 }
1351
1352 return (JSObject *)result;
1353
1355}
1356
1357
1358static BOOL JSNewNSDictionaryValue(JSContext *context, NSDictionary *dictionary, jsval *value)
1359{
1361
1362 JSObject *object = NULL;
1363 BOOL OK = YES;
1364
1365 if (value == NULL) return NO;
1366
1367 // NOTE: should be called within a local root scope or have *value be a set root for GC reasons.
1368 if (!JS_EnterLocalRootScope(context)) return NO;
1369
1370 object = JSObjectFromNSDictionary(context, dictionary);
1371 if (object == NULL)
1372 {
1373 *value = JSVAL_VOID;
1374 OK = NO;
1375 }
1376 else
1377 {
1378 *value = OBJECT_TO_JSVAL(object);
1379 }
1380
1381 JS_LeaveLocalRootScopeWithResult(context, *value);
1382 return OK;
1383
1385}
1386
1387
1388@implementation NSObject (OOJavaScriptConversion)
1389
1390- (jsval) oo_jsValueInContext:(JSContext *)context
1391{
1392 return JSVAL_VOID;
1393}
1394
1395
1396- (NSString *) oo_jsClassName
1397{
1398 return nil;
1399}
1400
1401
1402- (NSString *) oo_jsDescription
1403{
1404 return [self oo_jsDescriptionWithClassName:[self oo_jsClassName]];
1405}
1406
1407
1408- (NSString *) oo_jsDescriptionWithClassName:(NSString *)className
1409{
1411
1412 NSString *components = nil;
1413 NSString *description = nil;
1414
1415 components = [self descriptionComponents];
1416 if (className == nil) className = [[self class] description];
1417
1418 if (components != nil)
1419 {
1420 description = [NSString stringWithFormat:@"[%@ %@]", className, components];
1421 }
1422 else
1423 {
1424 description = [NSString stringWithFormat:@"[object %@]", className];
1425 }
1426
1427 return description;
1428
1430}
1431
1432
1433- (void) oo_clearJSSelf:(JSObject *)selfVal
1434{
1435
1436}
1437
1438@end
1439
1440
1441JSObject *OOJSObjectFromNativeObject(JSContext *context, id object)
1442{
1443 jsval value = OOJSValueFromNativeObject(context, object);
1444 JSObject *result = NULL;
1445 if (JS_ValueToObject(context, value, &result)) return result;
1446 return NULL;
1447}
1448
1449
1450@implementation OOJSValue
1451
1452+ (id) valueWithJSValue:(jsval)value inContext:(JSContext *)context
1453{
1455
1456 return [[[self alloc] initWithJSValue:value inContext:context] autorelease];
1457
1459}
1460
1461
1462+ (id) valueWithJSObject:(JSObject *)object inContext:(JSContext *)context
1463{
1465
1466 return [[[self alloc] initWithJSObject:object inContext:context] autorelease];
1467
1469}
1470
1471
1472- (id) initWithJSValue:(jsval)value inContext:(JSContext *)context
1473{
1475
1476 self = [super init];
1477 if (self != nil)
1478 {
1479 BOOL tempCtxt = NO;
1480 if (context == NULL)
1481 {
1482 context = OOJSAcquireContext();
1483 tempCtxt = YES;
1484 }
1485
1486 _val = value;
1487 if (!JSVAL_IS_VOID(_val))
1488 {
1489 JS_AddNamedValueRoot(context, &_val, "OOJSValue");
1490
1491 [[NSNotificationCenter defaultCenter] addObserver:self
1492 selector:@selector(deleteJSValue)
1493 name:kOOJavaScriptEngineWillResetNotification
1495 }
1496
1497 if (tempCtxt) OOJSRelinquishContext(context);
1498 }
1499 return self;
1500
1502}
1503
1504
1505- (id) initWithJSObject:(JSObject *)object inContext:(JSContext *)context
1506{
1507 return [self initWithJSValue:OBJECT_TO_JSVAL(object) inContext:context];
1508}
1509
1510
1511- (void) deleteJSValue
1512{
1513 if (!JSVAL_IS_VOID(_val))
1514 {
1515 JSContext *context = OOJSAcquireContext();
1516 JS_RemoveValueRoot(context, &_val);
1517 OOJSRelinquishContext(context);
1518
1519 _val = JSVAL_VOID;
1520 [[NSNotificationCenter defaultCenter] removeObserver:self
1521 name:kOOJavaScriptEngineWillResetNotification
1523 }
1524}
1525
1526
1527- (void) dealloc
1528{
1529 [self deleteJSValue];
1530 [super dealloc];
1531}
1532
1533
1534- (jsval) oo_jsValueInContext:(JSContext *)context
1535{
1536 return _val;
1537}
1538
1539@end
1540
1541
1542void OOJSStrLiteralCachePRIVATE(const char *string, jsval *strCache, BOOL *inited)
1543{
1544 NSCParameterAssert(string != NULL && strCache != NULL && inited != NULL && !*inited);
1545
1546 JSContext *context = OOJSAcquireContext();
1547
1548 JSString *jsString = JS_InternString(context, string);
1549 if (EXPECT_NOT(string == NULL))
1550 {
1551 [NSException raise:NSGenericException format:@"Failed to initialize JavaScript string literal cache for \"%@\".", [[NSString stringWithUTF8String:string] escapedForJavaScriptLiteral]];
1552 }
1553
1554 *strCache = STRING_TO_JSVAL(jsString);
1555 *inited = YES;
1556
1557 OOJSRelinquishContext(context);
1558}
1559
1560
1561NSString *OOStringFromJSString(JSContext *context, JSString *string)
1562{
1564
1565 if (EXPECT_NOT(string == NULL)) return nil;
1566
1567 size_t length;
1568 const jschar *chars = JS_GetStringCharsAndLength(context, string, &length);
1569
1570 if (EXPECT(chars != NULL))
1571 {
1572 return [NSString stringWithCharacters:chars length:length];
1573 }
1574 else
1575 {
1576 return nil;
1577 }
1578
1580}
1581
1582
1583NSString *OOStringFromJSValueEvenIfNull(JSContext *context, jsval value)
1584{
1586
1587 NSCParameterAssert(context != NULL && JS_IsInRequest(context));
1588
1589 JSString *string = JS_ValueToString(context, value); // Calls the value's toString method if needed.
1590 return OOStringFromJSString(context, string);
1591
1593}
1594
1595
1596NSString *OOStringFromJSValue(JSContext *context, jsval value)
1597{
1599
1600 if (EXPECT(!JSVAL_IS_NULL(value) && !JSVAL_IS_VOID(value)))
1601 {
1602 return OOStringFromJSValueEvenIfNull(context, value);
1603 }
1604 return nil;
1605
1607}
1608
1609
1610NSString *OOStringFromJSPropertyIDAndSpec(JSContext *context, jsid propID, JSPropertySpec *propertySpec)
1611{
1612 if (JSID_IS_STRING(propID))
1613 {
1614 return OOStringFromJSString(context, JSID_TO_STRING(propID));
1615 }
1616 else if (JSID_IS_INT(propID) && propertySpec != NULL)
1617 {
1618 int tinyid = JSID_TO_INT(propID);
1619
1620 while (propertySpec->name != NULL)
1621 {
1622 if (propertySpec->tinyid == tinyid) return [NSString stringWithUTF8String:propertySpec->name];
1623 propertySpec++;
1624 }
1625 }
1626
1627 jsval value;
1628 if (!JS_IdToValue(context, propID, &value)) return @"unknown";
1629 return OOStringFromJSString(context, JS_ValueToString(context, value));
1630}
1631
1632
1633static NSString *DescribeValue(JSContext *context, jsval value, BOOL abbreviateObjects, BOOL recursing)
1634{
1636
1637 NSCParameterAssert(context != NULL && JS_IsInRequest(context));
1638
1639 if (OOJSValueIsFunction(context, value))
1640 {
1641 JSString *name = JS_GetFunctionId(JS_ValueToFunction(context, value));
1642 if (name != NULL) return [NSString stringWithFormat:@"function %@", OOStringFromJSString(context, name)];
1643 else return @"function";
1644 }
1645
1646 NSString *result = nil;
1647 JSClass *class = NULL;
1649
1650 if (JSVAL_IS_OBJECT(value) && !JSVAL_IS_NULL(value))
1651 {
1652 class = OOJSGetClass(context, JSVAL_TO_OBJECT(value));
1653 }
1654
1655 // Convert String objects to strings.
1656 if (class == [jsEng stringClass])
1657 {
1658 value = STRING_TO_JSVAL(JS_ValueToString(context, value));
1659 }
1660
1661 if (JSVAL_IS_STRING(value))
1662 {
1663 enum { kMaxLength = 200 };
1664
1665 JSString *string = JSVAL_TO_STRING(value);
1666 size_t length;
1667 const jschar *chars = JS_GetStringCharsAndLength(context, string, &length);
1668
1669 result = [NSString stringWithCharacters:chars length:MIN(length, (size_t)kMaxLength)];
1670 result = [NSString stringWithFormat:@"\"%@%@\"", [result escapedForJavaScriptLiteral], (length > kMaxLength) ? @"..." : @""];
1671 }
1672 else if (class == [jsEng arrayClass])
1673 {
1674 // Descibe up to four elements of an array.
1675 jsuint count;
1676 JSObject *obj = JSVAL_TO_OBJECT(value);
1677 if (JS_GetArrayLength(context, obj, &count))
1678 {
1679 if (!recursing)
1680 {
1681 NSMutableString *arrayDesc = [NSMutableString stringWithString:@"["];
1682 jsuint i, effectiveCount = MIN(count, (jsuint)4);
1683 for (i = 0; i < effectiveCount; i++)
1684 {
1685 jsval item;
1686 NSString *itemDesc = @"?";
1687 if (JS_GetElement(context, obj, i, &item))
1688 {
1689 itemDesc = DescribeValue(context, item, YES /* always abbreviate objects in arrays */, YES);
1690 }
1691 if (i != 0) [arrayDesc appendString:@", "];
1692 [arrayDesc appendString:itemDesc];
1693 }
1694 if (effectiveCount != count)
1695 {
1696 [arrayDesc appendFormat:@", ... <%u items total>]", count];
1697 }
1698 else
1699 {
1700 [arrayDesc appendString:@"]"];
1701 }
1702
1703 result = arrayDesc;
1704 }
1705 else
1706 {
1707 result = [NSString stringWithFormat:@"[<%u items>]", count];
1708 }
1709 }
1710 else
1711 {
1712 result = @"[...]";
1713 }
1714
1715 }
1716
1717 if (result == nil)
1718 {
1719 result = OOStringFromJSValueEvenIfNull(context, value);
1720
1721 if (abbreviateObjects && class == [jsEng objectClass] && [result isEqualToString:@"[object Object]"])
1722 {
1723 result = @"{...}";
1724 }
1725
1726 if (result == nil) result = @"?";
1727 }
1728
1729 return result;
1730
1732}
1733
1734
1735NSString *OOJSDescribeValue(JSContext *context, jsval value, BOOL abbreviateObjects)
1736{
1737 return DescribeValue(context, value, abbreviateObjects, NO);
1738}
1739
1740
1741@implementation NSString (OOJavaScriptExtensions)
1742
1743+ (NSString *) stringWithJavaScriptParameters:(jsval *)params count:(uintN)count inContext:(JSContext *)context
1744{
1746
1747 if (params == NULL && count != 0) return nil;
1748
1749 uintN i;
1750 NSMutableString *result = [NSMutableString stringWithString:@"("];
1751
1752 for (i = 0; i < count; ++i)
1753 {
1754 if (i != 0) [result appendString:@", "];
1755 [result appendString:OOJSDescribeValue(context, params[i], NO)];
1756 }
1757
1758 [result appendString:@")"];
1759 return result;
1760
1762}
1763
1764
1765- (jsval) oo_jsValueInContext:(JSContext *)context
1766{
1768
1769 size_t length = [self length];
1770 unichar *buffer = NULL;
1771 JSString *string = NULL;
1772
1773 if (length == 0)
1774 {
1775 jsval result = JS_GetEmptyStringValue(context);
1776 return result;
1777 }
1778 else
1779 {
1780 buffer = malloc(length * sizeof *buffer);
1781 if (buffer == NULL) return JSVAL_VOID;
1782
1783 [self getCharacters:buffer];
1784
1785 string = JS_NewUCStringCopyN(context, buffer, length);
1786
1787 free(buffer);
1788 return STRING_TO_JSVAL(string);
1789 }
1790
1792}
1793
1794
1795+ (NSString *) concatenationOfStringsFromJavaScriptValues:(jsval *)values count:(size_t)count separator:(NSString *)separator inContext:(JSContext *)context
1796{
1798
1799 size_t i;
1800 NSMutableString *result = nil;
1801 NSString *element = nil;
1802
1803 if (count < 1) return nil;
1804 if (values == NULL) return NULL;
1805
1806 for (i = 0; i != count; ++i)
1807 {
1808 element = OOStringFromJSValueEvenIfNull(context, values[i]);
1809 if (result == nil) result = [[element mutableCopy] autorelease];
1810 else
1811 {
1812 if (separator != nil) [result appendString:separator];
1813 [result appendString:element];
1814 }
1815 }
1816
1817 return result;
1818
1820}
1821
1822
1823- (NSString *)escapedForJavaScriptLiteral
1824{
1826
1827 NSMutableString *result = nil;
1828 NSUInteger i, length;
1829 unichar c;
1830 NSAutoreleasePool *pool = nil;
1831
1832 length = [self length];
1833 result = [NSMutableString stringWithCapacity:length];
1834
1835 // Not hugely efficient.
1836 pool = [[NSAutoreleasePool alloc] init];
1837 for (i = 0; i != length; ++i)
1838 {
1839 c = [self characterAtIndex:i];
1840 switch (c)
1841 {
1842 case '\\':
1843 [result appendString:@"\\\\"];
1844 break;
1845
1846 case '\b':
1847 [result appendString:@"\\b"];
1848 break;
1849
1850 case '\f':
1851 [result appendString:@"\\f"];
1852 break;
1853
1854 case '\n':
1855 [result appendString:@"\\n"];
1856 break;
1857
1858 case '\r':
1859 [result appendString:@"\\r"];
1860 break;
1861
1862 case '\t':
1863 [result appendString:@"\\t"];
1864 break;
1865
1866 case '\v':
1867 [result appendString:@"\\v"];
1868 break;
1869
1870 case '\'':
1871 [result appendString:@"\\\'"];
1872 break;
1873
1874 case '\"':
1875 [result appendString:@"\\\""];
1876 break;
1877
1878 default:
1879 [result appendString:[NSString stringWithCharacters:&c length:1]];
1880 }
1881 }
1882 [pool release];
1883 return result;
1884
1886}
1887
1888
1889- (NSString *) oo_jsClassName
1890{
1891 return @"String";
1892}
1893
1894@end
1895
1896
1897@implementation NSArray (OOJavaScriptConversion)
1898
1899- (jsval)oo_jsValueInContext:(JSContext *)context
1900{
1901 jsval value = JSVAL_VOID;
1902 JSNewNSArrayValue(context, self, &value);
1903 return value;
1904}
1905
1906@end
1907
1908@implementation OONativeVector (OOJavaScriptConversion)
1909
1910- (jsval)oo_jsValueInContext:(JSContext *)context
1911{
1912 jsval value = JSVAL_VOID;
1913 VectorToJSValue(context, v, &value);
1914 return value;
1915}
1916
1917@end
1918
1919
1920@implementation NSDictionary (OOJavaScriptConversion)
1921
1922- (jsval)oo_jsValueInContext:(JSContext *)context
1923{
1924 jsval value = JSVAL_VOID;
1925 JSNewNSDictionaryValue(context, self, &value);
1926 return value;
1927}
1928
1929@end
1930
1931
1932@implementation NSNumber (OOJavaScriptConversion)
1933
1934- (jsval)oo_jsValueInContext:(JSContext *)context
1935{
1937
1938 jsval result;
1939 BOOL isFloat = NO;
1940 long long longLongValue;
1941
1942 isFloat = [self oo_isFloatingPointNumber];
1943 if (!isFloat)
1944 {
1945 longLongValue = [self longLongValue];
1946 if (longLongValue < (long long)JSVAL_INT_MIN || (long long)JSVAL_INT_MAX < longLongValue)
1947 {
1948 // values outside JSVAL_INT range are returned as doubles.
1949 isFloat = YES;
1950 }
1951 }
1952
1953 if (isFloat)
1954 {
1955 if (!JS_NewNumberValue(context, [self doubleValue], &result)) result = JSVAL_VOID;
1956 }
1957 else
1958 {
1959 result = INT_TO_JSVAL((int32_t)longLongValue);
1960 }
1961
1962 return result;
1963
1965}
1966
1967
1968- (NSString *) oo_jsClassName
1969{
1970 return @"Number";
1971}
1972
1973@end
1974
1975
1976@implementation NSNull (OOJavaScriptConversion)
1977
1978- (jsval)oo_jsValueInContext:(JSContext *)context
1979{
1980 return JSVAL_NULL;
1981}
1982
1983@end
1984
1985
1986JSBool OOJSUnconstructableConstruct(JSContext *context, uintN argc, jsval *vp)
1987{
1988 OOJS_NATIVE_ENTER(context)
1989
1990 JSFunction *function = JS_ValueToFunction(context, JS_CALLEE(context, vp));
1991 NSString *name = OOStringFromJSString(context, JS_GetFunctionId(function));
1992
1993 OOJSReportError(context, @"%@ cannot be used as a constructor.", name);
1994 return NO;
1995
1997}
1998
1999
2000void OOJSObjectWrapperFinalize(JSContext *context, JSObject *this)
2001{
2003
2004 id object = JS_GetPrivate(context, this);
2005 if (object != nil)
2006 {
2007 [[object weakRefUnderlyingObject] oo_clearJSSelf:this];
2008 [object release];
2009 JS_SetPrivate(context, this, nil);
2010 }
2011
2013}
2014
2015
2016JSBool OOJSObjectWrapperToString(JSContext *context, uintN argc, jsval *vp)
2017{
2018 OOJS_NATIVE_ENTER(context)
2019
2020 id object = nil;
2021 NSString *description = nil;
2022 JSClass *jsClass = NULL;
2023
2024 object = OOJSNativeObjectFromJSObject(context, OOJS_THIS);
2025 if (object != nil)
2026 {
2027 description = [object oo_jsDescription];
2028 if (description == nil) description = [object description];
2029 }
2030 if (description == nil)
2031 {
2032 jsClass = OOJSGetClass(context, OOJS_THIS);
2033 if (jsClass != NULL)
2034 {
2035 description = [NSString stringWithFormat:@"[object %@]", [NSString stringWithUTF8String:jsClass->name]];
2036 }
2037 }
2038 if (description == nil) description = @"[object]";
2039
2040 OOJS_RETURN_OBJECT(description);
2041
2043}
2044
2045
2046BOOL JSFunctionPredicate(Entity *entity, void *parameter)
2047{
2049
2050 JSFunctionPredicateParameter *param = parameter;
2051 jsval args[1];
2052 jsval rval = JSVAL_VOID;
2053 JSBool result = NO;
2054
2055 NSCParameterAssert(entity != nil && param != NULL);
2056 NSCParameterAssert(param->context != NULL && JS_IsInRequest(param->context));
2057 NSCParameterAssert(OOJSValueIsFunction(param->context, param->function));
2058
2059 if (EXPECT_NOT(param->errorFlag)) return NO;
2060
2061 args[0] = [entity oo_jsValueInContext:param->context]; // entity is required to be non-nil (asserted above), so oo_jsValueInContext: is safe.
2062
2065 BOOL success = JS_CallFunctionValue(param->context, param->jsThis, param->function, 1, args, &rval);
2068
2069 if (success)
2070 {
2071 if (!JS_ValueToBoolean(param->context, rval, &result)) result = NO;
2072 if (JS_IsExceptionPending(param->context))
2073 {
2074 JS_ReportPendingException(param->context);
2075 param->errorFlag = YES;
2076 }
2077 }
2078 else
2079 {
2080 param->errorFlag = YES;
2081 }
2082
2083 return result;
2084
2086}
2087
2088
2089BOOL JSEntityIsJavaScriptVisiblePredicate(Entity *entity, void *parameter)
2090{
2092
2093 return [entity isVisibleToScripts];
2094
2096}
2097
2098
2100{
2102
2103 if (![entity isVisibleToScripts]) return NO;
2104 if ([entity isShip])
2105 {
2106 if ([entity isSubEntity]) return NO;
2107 if ([entity status] == STATUS_COCKPIT_DISPLAY) return NO; // Demo ship
2108 return YES;
2109 }
2110 else if ([entity isPlanet])
2111 {
2112 switch ([(OOPlanetEntity *)entity planetType])
2113 {
2114 case STELLAR_TYPE_MOON:
2116 case STELLAR_TYPE_SUN:
2117 return YES;
2118
2119#if !NEW_PLANETS
2121#endif
2123 return NO;
2124 }
2125 }
2126
2127 return YES; // would happen if we added a new script-visible class
2128
2130}
2131
2132
2133BOOL JSEntityIsDemoShipPredicate(Entity *entity, void *parameter)
2134{
2135 return ([entity isVisibleToScripts] && [entity isShip] && [entity status] == STATUS_COCKPIT_DISPLAY && ![entity isSubEntity]);
2136}
2137
2138static NSMapTable *sRegisteredSubClasses;
2139
2140void OOJSRegisterSubclass(JSClass *subclass, JSClass *superclass)
2141{
2142 NSCParameterAssert(subclass != NULL && superclass != NULL);
2143
2144 if (sRegisteredSubClasses == NULL)
2145 {
2146 sRegisteredSubClasses = NSCreateMapTable(NSNonOwnedPointerMapKeyCallBacks, NSNonOwnedPointerMapValueCallBacks, 0);
2147 }
2148
2149 NSCAssert(NSMapGet(sRegisteredSubClasses, subclass) == NULL, @"A JS class cannot be registered as a subclass of multiple classes.");
2150
2151 NSMapInsertKnownAbsent(sRegisteredSubClasses, subclass, superclass);
2152}
2153
2154
2155static void UnregisterSubclasses(void)
2156{
2157 NSFreeMapTable(sRegisteredSubClasses);
2158 sRegisteredSubClasses = NULL;
2159}
2160
2161
2162BOOL OOJSIsSubclass(JSClass *putativeSubclass, JSClass *superclass)
2163{
2164 NSCParameterAssert(putativeSubclass != NULL && superclass != NULL);
2165 NSCAssert(sRegisteredSubClasses != NULL, @"OOJSIsSubclass() called before any subclasses registered (disallowed for hot path efficiency).");
2166
2167 do
2168 {
2169 if (putativeSubclass == superclass) return YES;
2170
2171 putativeSubclass = NSMapGet(sRegisteredSubClasses, putativeSubclass);
2172 }
2173 while (putativeSubclass != NULL);
2174
2175 return NO;
2176}
2177
2178
2179BOOL OOJSObjectGetterImplPRIVATE(JSContext *context, JSObject *object, JSClass *requiredJSClass,
2180#ifndef NDEBUG
2181 Class requiredObjCClass, const char *name,
2182#endif
2183 id *outObject)
2184{
2185#ifndef NDEBUG
2186 OOJS_PROFILE_ENTER_NAMED(name)
2187 NSCParameterAssert(requiredObjCClass != Nil);
2188 NSCParameterAssert(context != NULL && object != NULL && requiredJSClass != NULL && outObject != NULL);
2189#else
2191#endif
2192
2193 /*
2194 Ensure it's a valid type of JS object. This is absolutely necessary,
2195 because if we don't check it we'll crash trying to get the private
2196 field of something that isn't an ObjC object wrapper - for example,
2197 Ship.setAI.call(new Vector3D, "") is valid JavaScript.
2198
2199 Alternatively, we could abuse JSCLASS_PRIVATE_IS_NSISUPPORTS as a
2200 flag for ObjC object wrappers (SpiderMonkey only uses it internally
2201 in a debug function we don't use), but we'd still need to do an
2202 Objective-C class test, and I don't think that's any faster.
2203 TODO: profile.
2204 */
2205 JSClass *actualClass = OOJSGetClass(context, object);
2206 if (EXPECT_NOT(!OOJSIsSubclass(actualClass, requiredJSClass)))
2207 {
2208 OOJSReportError(context, @"Native method expected %s, got %@.", requiredJSClass->name, OOStringFromJSValue(context, OBJECT_TO_JSVAL(object)));
2209 return NO;
2210 }
2211 NSCAssert(actualClass->flags & JSCLASS_HAS_PRIVATE, @"Native object accessor requires JS class with private storage.");
2212
2213 // Get the underlying object.
2214 *outObject = [(id)JS_GetPrivate(context, object) weakRefUnderlyingObject];
2215
2216#ifndef NDEBUG
2217 // Double-check that the underlying object is of the expected ObjC class.
2218 if (EXPECT_NOT(*outObject != nil && ![*outObject isKindOfClass:requiredObjCClass]))
2219 {
2220 OOJSReportError(context, @"Native method expected %@ from %s and got correct JS type but incorrect native object %@", requiredObjCClass, requiredJSClass->name, *outObject);
2221 return NO;
2222 }
2223#endif
2224
2225 return YES;
2226
2228}
2229
2230
2231NSDictionary *OOJSDictionaryFromJSValue(JSContext *context, jsval value)
2232{
2234
2235 JSObject *object = NULL;
2236 if (EXPECT_NOT(!JS_ValueToObject(context, value, &object) || object == NULL))
2237 {
2238 return nil;
2239 }
2240 return OOJSDictionaryFromJSObject(context, object);
2241
2243}
2244
2245
2246NSDictionary *OOJSDictionaryFromJSObject(JSContext *context, JSObject *object)
2247{
2249
2250 JSIdArray *ids = NULL;
2251 jsint i;
2252 NSMutableDictionary *result = nil;
2253 jsval value = JSVAL_VOID;
2254 id objKey = nil;
2255 id objValue = nil;
2256
2257 ids = JS_Enumerate(context, object);
2258 if (EXPECT_NOT(ids == NULL))
2259 {
2260 return nil;
2261 }
2262
2263 result = [NSMutableDictionary dictionaryWithCapacity:ids->length];
2264 for (i = 0; i != ids->length; ++i)
2265 {
2266 jsid thisID = ids->vector[i];
2267
2268 if (JSID_IS_STRING(thisID))
2269 {
2270 objKey = OOStringFromJSString(context, JSID_TO_STRING(thisID));
2271 }
2272 else if (JSID_IS_INT(thisID))
2273 {
2274 /* this causes problems with native functions which expect string keys
2275 * e.g. in mission.runScreen with the 'choices' parameter
2276 * should this instead be making the objKey a string?
2277 * is there anything that relies on the current behaviour?
2278 * - CIM 15/2/13 */
2279 objKey = [NSNumber numberWithInt:JSID_TO_INT(thisID)];
2280 }
2281 else
2282 {
2283 objKey = nil;
2284 }
2285
2286 value = JSVAL_VOID;
2287 if (objKey != nil && !JS_LookupPropertyById(context, object, thisID, &value)) value = JSVAL_VOID;
2288
2289 if (objKey != nil && !JSVAL_IS_VOID(value))
2290 {
2291 objValue = OOJSNativeObjectFromJSValue(context, value);
2292 if (objValue != nil)
2293 {
2294 [result setObject:objValue forKey:objKey];
2295 }
2296 }
2297 }
2298
2299 JS_DestroyIdArray(context, ids);
2300 return result;
2301
2303}
2304
2305
2306NSDictionary *OOJSDictionaryFromStringTable(JSContext *context, jsval tableValue)
2307{
2309
2310 JSObject *tableObject = NULL;
2311 JSIdArray *ids;
2312 jsint i;
2313 NSMutableDictionary *result = nil;
2314 jsval value = JSVAL_VOID;
2315 id objKey = nil;
2316 id objValue = nil;
2317
2318 if (EXPECT_NOT(JSVAL_IS_NULL(tableValue) || !JS_ValueToObject(context, tableValue, &tableObject)))
2319 {
2320 return nil;
2321 }
2322
2323 ids = JS_Enumerate(context, tableObject);
2324 if (EXPECT_NOT(ids == NULL))
2325 {
2326 return nil;
2327 }
2328
2329 result = [NSMutableDictionary dictionaryWithCapacity:ids->length];
2330 for (i = 0; i != ids->length; ++i)
2331 {
2332 jsid thisID = ids->vector[i];
2333
2334 if (JSID_IS_STRING(thisID))
2335 {
2336 objKey = OOStringFromJSString(context, JSID_TO_STRING(thisID));
2337 }
2338 else
2339 {
2340 objKey = nil;
2341 }
2342
2343 value = JSVAL_VOID;
2344 if (objKey != nil && !JS_LookupPropertyById(context, tableObject, thisID, &value)) value = JSVAL_VOID;
2345
2346 if (objKey != nil && !JSVAL_IS_VOID(value))
2347 {
2348 objValue = OOStringFromJSValueEvenIfNull(context, value);
2349
2350 if (objValue != nil)
2351 {
2352 [result setObject:objValue forKey:objKey];
2353 }
2354 }
2355 }
2356
2357 JS_DestroyIdArray(context, ids);
2358 return result;
2359
2361}
2362
2363
2364static NSMutableDictionary *sObjectConverters;
2365
2366
2367id OOJSNativeObjectFromJSValue(JSContext *context, jsval value)
2368{
2370
2371 if (JSVAL_IS_NULL(value) || JSVAL_IS_VOID(value)) return nil;
2372
2373 if (JSVAL_IS_INT(value))
2374 {
2375 return [NSNumber numberWithInt:JSVAL_TO_INT(value)];
2376 }
2377 if (JSVAL_IS_DOUBLE(value))
2378 {
2379 return [NSNumber numberWithDouble:JSVAL_TO_DOUBLE(value)];
2380 }
2381 if (JSVAL_IS_BOOLEAN(value))
2382 {
2383 return [NSNumber numberWithBool:JSVAL_TO_BOOLEAN(value)];
2384 }
2385 if (JSVAL_IS_STRING(value))
2386 {
2387 return OOStringFromJSValue(context, value);
2388 }
2389 if (JSVAL_IS_OBJECT(value))
2390 {
2391 return OOJSNativeObjectFromJSObject(context, JSVAL_TO_OBJECT(value));
2392 }
2393 return nil;
2394
2396}
2397
2398
2399id OOJSNativeObjectFromJSObject(JSContext *context, JSObject *tableObject)
2400{
2402
2403 NSValue *wrappedClass = nil;
2404 NSValue *wrappedConverter = nil;
2405 OOJSClassConverterCallback converter = NULL;
2406 JSClass *class = NULL;
2407
2408 if (tableObject == NULL) return nil;
2409
2410 class = OOJSGetClass(context, tableObject);
2411 wrappedClass = [NSValue valueWithPointer:class];
2412 if (wrappedClass != nil) wrappedConverter = [sObjectConverters objectForKey:wrappedClass];
2413 if (wrappedConverter != nil)
2414 {
2415 converter = [wrappedConverter pointerValue];
2416 return converter(context, tableObject);
2417 }
2418 return nil;
2419
2421}
2422
2423
2424id OOJSNativeObjectOfClassFromJSValue(JSContext *context, jsval value, Class requiredClass)
2425{
2426 id result = OOJSNativeObjectFromJSValue(context, value);
2427 if (![result isKindOfClass:requiredClass]) result = nil;
2428 return result;
2429}
2430
2431
2432id OOJSNativeObjectOfClassFromJSObject(JSContext *context, JSObject *object, Class requiredClass)
2433{
2434 id result = OOJSNativeObjectFromJSObject(context, object);
2435 if (![result isKindOfClass:requiredClass]) result = nil;
2436 return result;
2437}
2438
2439
2440id OOJSBasicPrivateObjectConverter(JSContext *context, JSObject *object)
2441{
2442 id result;
2443
2444 /* This will do the right thing - for non-OOWeakReferences,
2445 weakRefUnderlyingObject returns the object itself. For nil, of course,
2446 it returns nil.
2447 */
2448 result = JS_GetPrivate(context, object);
2449 return [result weakRefUnderlyingObject];
2450}
2451
2452
2454{
2455 NSValue *wrappedClass = nil;
2456 NSValue *wrappedConverter = nil;
2457
2458 if (theClass == NULL) return;
2459 if (sObjectConverters == nil) sObjectConverters = [[NSMutableDictionary alloc] init];
2460
2461 wrappedClass = [NSValue valueWithPointer:theClass];
2462 if (converter != NULL)
2463 {
2464 wrappedConverter = [NSValue valueWithPointer:converter];
2465 [sObjectConverters setObject:wrappedConverter forKey:wrappedClass];
2466 }
2467 else
2468 {
2469 [sObjectConverters removeObjectForKey:wrappedClass];
2470 }
2471}
2472
2473
2475{
2477}
2478
2479
2480static id JSArrayConverter(JSContext *context, JSObject *array)
2481{
2482 jsuint i, count;
2483 id *values = NULL;
2484 jsval value = JSVAL_VOID;
2485 id object = nil;
2486 NSArray *result = nil;
2487
2488 // Convert a JS array to an NSArray by calling OOJSNativeObjectFromJSValue() on all its elements.
2489 if (!JS_IsArrayObject(context, array)) return nil;
2490 if (!JS_GetArrayLength(context, array, &count)) return nil;
2491
2492 if (count == 0) return [NSArray array];
2493
2494 values = calloc(count, sizeof *values);
2495 if (values == NULL) return nil;
2496
2497 for (i = 0; i != count; ++i)
2498 {
2499 value = JSVAL_VOID;
2500 if (!JS_GetElement(context, array, i, &value)) value = JSVAL_VOID;
2501
2502 object = OOJSNativeObjectFromJSValue(context, value);
2503 if (object == nil) object = [NSNull null];
2504 values[i] = object;
2505 }
2506
2507 result = [NSArray arrayWithObjects:values count:count];
2508 free(values);
2509 return result;
2510}
2511
2512
2513static id JSStringConverter(JSContext *context, JSObject *object)
2514{
2515 return OOStringFromJSValue(context, OBJECT_TO_JSVAL(object));
2516}
2517
2518
2519static id JSNumberConverter(JSContext *context, JSObject *object)
2520{
2521 jsdouble value;
2522 if (JS_ValueToNumber(context, OBJECT_TO_JSVAL(object), &value))
2523 {
2524 return [NSNumber numberWithDouble:value];
2525 }
2526 return nil;
2527}
2528
2529
2530static id JSBooleanConverter(JSContext *context, JSObject *object)
2531{
2532 /* Fun With JavaScript: Boolean(false) is a truthy value, since it's a
2533 non-null object. JS_ValueToBoolean() therefore reports true.
2534 However, Boolean objects are transformed to numbers sanely, so this
2535 works.
2536 */
2537 jsdouble value;
2538 if (JS_ValueToNumber(context, OBJECT_TO_JSVAL(object), &value))
2539 {
2540 return [NSNumber numberWithBool:(value != 0)];
2541 }
2542 return nil;
2543}
#define DESTROY(x)
Definition OOCocoa.h:75
#define foreachkey(VAR, DICT)
Definition OOCocoa.h:353
void OOConstToJSStringDestroy(void)
void OOConstToJSStringInit(JSContext *context)
#define EXPECT_NOT(x)
#define EXPECT(x)
void InitOOJSClock(JSContext *context, JSObject *global)
Definition OOJSClock.m:109
void InitOOJSDock(JSContext *context, JSObject *global)
Definition OOJSDock.m:94
#define OOJS_PROFILE_EXIT
#define OOJS_PROFILE_EXIT_VOID
#define OOJS_NATIVE_ENTER(cx)
#define OOJS_NATIVE_EXIT
#define OOJS_PROFILE_ENTER
#define OOJS_PROFILE_EXIT_JSVAL
#define OOJSStopTimeLimiter()
#define OOJSStartTimeLimiter()
void OOJSTimeManagementInit(OOJavaScriptEngine *engine, JSRuntime *runtime)
void InitOOJSEntity(JSContext *context, JSObject *global)
Definition OOJSEntity.m:139
void InitOOJSEquipmentInfo(JSContext *context, JSObject *global)
void InitOOJSExhaustPlume(JSContext *context, JSObject *global)
void InitOOJSFlasher(JSContext *context, JSObject *global)
Definition OOJSFlasher.m:96
void InitOOJSFont(JSContext *context, JSObject *global)
Definition OOJSFont.m:38
void InitOOJSFrameCallbacks(JSContext *context, JSObject *global)
void OOJSFrameCallbacksRemoveAll(void)
void CreateOOJSGlobal(JSContext *context, JSObject **outGlobal)
Definition OOJSGlobal.m:162
void SetUpOOJSGlobal(JSContext *context, JSObject *global)
Definition OOJSGlobal.m:173
void InitOOJSManifest(JSContext *context, JSObject *global)
void InitOOJSMissionVariables(JSContext *context, JSObject *global)
void MissionRunCallback(void)
void InitOOJSMission(JSContext *context, JSObject *global)
void InitOOJSOolite(JSContext *context, JSObject *global)
Definition OOJSOolite.m:110
void InitOOJSPlanet(JSContext *context, JSObject *global)
Definition OOJSPlanet.m:99
void InitOOJSPlayerShip(JSContext *context, JSObject *global)
void InitOOJSPlayer(JSContext *context, JSObject *global)
Definition OOJSPlayer.m:163
void InitOOJSQuaternion(JSContext *context, JSObject *global)
void InitOOJSScript(JSContext *context, JSObject *global)
Definition OOJSScript.m:662
void InitOOJSShipGroup(JSContext *context, JSObject *global)
void InitOOJSShip(JSContext *context, JSObject *global)
Definition OOJSShip.m:587
void InitOOJSSoundSource(JSContext *context, JSObject *global)
void InitOOJSSound(JSContext *context, JSObject *global)
Definition OOJSSound.m:104
void InitOOJSSpecialFunctions(JSContext *context, JSObject *global)
void InitOOJSStation(JSContext *context, JSObject *global)
void InitOOJSSun(JSContext *context, JSObject *global)
Definition OOJSSun.m:91
void InitOOJSSystemInfo(JSContext *context, JSObject *global)
void InitOOJSSystem(JSContext *context, JSObject *global)
Definition OOJSSystem.m:227
void InitOOJSTimer(JSContext *context, JSObject *global)
Definition OOJSTimer.m:254
void InitOOJSVector(JSContext *context, JSObject *global)
Definition OOJSVector.m:153
BOOL VectorToJSValue(JSContext *context, Vector vector, jsval *outValue) NONNULL_FUNC
Definition OOJSVector.m:185
void InitOOJSVisualEffect(JSContext *context, JSObject *global)
void InitOOJSWaypoint(JSContext *context, JSObject *global)
void InitOOJSWorldScripts(JSContext *context, JSObject *global)
void InitOOJSWormhole(JSContext *context, JSObject *global)
void OOJSPauseTimeLimiter(void)
id OOJSNativeObjectFromJSValue(JSContext *context, jsval value)
jsid OOJSIDFromString(NSString *string)
BOOL JSEntityIsDemoShipPredicate(Entity *entity, void *parameter)
BOOL OOJSArgumentListGetNumberNoError(JSContext *context, uintN argc, jsval *argv, double *outNumber, uintN *outConsumed)
void OOJSReportWarning(JSContext *context, NSString *format,...)
#define OOJS_THIS
BOOL OOJSArgumentListGetNumber(JSContext *context, NSString *scriptClass, NSString *function, uintN argc, jsval *argv, double *outNumber, uintN *outConsumed)
JSBool OOJSObjectWrapperToString(JSContext *context, uintN argc, jsval *vp)
BOOL OOJSObjectGetterImplPRIVATE(JSContext *context, JSObject *object, JSClass *requiredJSClass, Class requiredObjCClass, const char *name, id *outObject)
BOOL JSEntityIsJavaScriptSearchablePredicate(Entity *entity, void *parameter)
void OOJSReportWarningWithArguments(JSContext *context, NSString *format, va_list args)
NSString * OOStringFromJSPropertyIDAndSpec(JSContext *context, jsid propID, JSPropertySpec *propertySpec)
#define JS_IsInRequest(context)
NSString * OOStringFromJSID(jsid propID)
void OOJSRegisterObjectConverter(JSClass *theClass, OOJSClassConverterCallback converter)
void OOJSSetWarningOrErrorStackSkip(unsigned skip)
BOOL JSFunctionPredicate(Entity *entity, void *parameter)
OOINLINE jsval OOJSValueFromNativeObject(JSContext *context, id object)
id OOJSNativeObjectFromJSObject(JSContext *context, JSObject *object)
void OOJSObjectWrapperFinalize(JSContext *context, JSObject *this)
#define OOJS_RETURN_OBJECT(o)
void OOJSReportBadPropertySelector(JSContext *context, JSObject *thisObj, jsid propID, JSPropertySpec *propertySpec)
NSDictionary * OOJSDictionaryFromJSValue(JSContext *context, jsval value)
void OOJSReportWarningForCaller(JSContext *context, NSString *scriptClass, NSString *function, NSString *format,...)
void OOJSReportErrorForCaller(JSContext *context, NSString *scriptClass, NSString *function, NSString *format,...)
NSDictionary * OOJSDictionaryFromJSObject(JSContext *context, JSObject *object)
void OOJSMarkConsoleEvalLocation(JSContext *context, JSStackFrame *stackFrame)
id OOJSNativeObjectOfClassFromJSValue(JSContext *context, jsval value, Class requiredClass)
NSString * OOStringFromJSValue(JSContext *context, jsval value)
JSBool OOJSUnconstructableConstruct(JSContext *context, uintN argc, jsval *vp)
OOINLINE BOOL OOJSValueIsFunction(JSContext *context, jsval value)
void OOJSRegisterSubclass(JSClass *subclass, JSClass *superclass)
JSObject * OOJSObjectFromNativeObject(JSContext *context, id object)
OOINLINE JSContext * OOJSAcquireContext(void)
NSString * OOJSDescribeValue(JSContext *context, jsval value, BOOL abbreviateObjects)
void OOJSReportError(JSContext *context, NSString *format,...)
#define JS_BeginRequest(context)
NSString * OOStringFromJSValueEvenIfNull(JSContext *context, jsval value)
OOINLINE void OOJSRelinquishContext(JSContext *context)
id(* OOJSClassConverterCallback)(JSContext *context, JSObject *object)
void OOJSReportBadPropertyValue(JSContext *context, JSObject *thisObj, jsid propID, JSPropertySpec *propertySpec, jsval value)
id OOJSBasicPrivateObjectConverter(JSContext *context, JSObject *object)
void OOJSReportBadArguments(JSContext *context, NSString *scriptClass, NSString *function, uintN argc, jsval *argv, NSString *message, NSString *expectedArgsDescription)
#define JS_EndRequest(context)
NSDictionary * OOJSDictionaryFromStringTable(JSContext *context, jsval value)
void OOJSReportErrorWithArguments(JSContext *context, NSString *format, va_list args)
void OOJSResumeTimeLimiter(void)
BOOL JSEntityIsJavaScriptVisiblePredicate(Entity *entity, void *parameter)
NSString * OOJSDescribeLocation(JSContext *context, JSStackFrame *stackFrame)
BOOL OOJSIsSubclass(JSClass *putativeSubclass, JSClass *superclass)
void OOJSDumpStack(JSContext *context)
NSString * OOStringFromJSString(JSContext *context, JSString *string)
OOINLINE JSClass * OOJSGetClass(JSContext *cx, JSObject *obj) ALWAYS_INLINE_FUNC
void OOJSStrLiteralCachePRIVATE(const char *string, jsval *strCache, BOOL *inited)
id OOJSNativeObjectOfClassFromJSObject(JSContext *context, JSObject *object, Class requiredClass)
static id JSNumberConverter(JSContext *context, JSObject *object)
static NSUInteger sConsoleEvalLineNo
static NSString * CallerPrefix(NSString *scriptClass, NSString *function)
static const char * sConsoleScriptName
static id JSStringConverter(JSContext *context, JSObject *object)
static void DumpVariable(JSContext *context, JSPropertyDesc *prop)
NSString *const kOOJavaScriptEngineWillResetNotification
void OOJSReportWrappedException(JSContext *context, id exception)
NSString *const kOOJavaScriptEngineDidResetNotification
#define OOJSENGINE_JSVERSION
static OOJavaScriptEngine * sSharedEngine
static id JSArrayConverter(JSContext *context, JSObject *object)
static void ReportJSError(JSContext *context, const char *message, JSErrorReport *report)
#define OOJSENGINE_CONTEXT_OPTIONS
void OOJSUnreachable(const char *function, const char *file, unsigned line)
static NSMapTable * sRegisteredSubClasses
static NSString * DescribeValue(JSContext *context, jsval value, BOOL abbreviateObjects, BOOL recursing)
static void UnregisterSubclasses(void)
static id JSBooleanConverter(JSContext *context, JSObject *object)
static JSObject * JSArrayFromNSArray(JSContext *context, NSArray *array)
static JSObject * JSObjectFromNSDictionary(JSContext *context, NSDictionary *dictionary)
static void GetLocationNameAndLine(JSContext *context, JSStackFrame *stackFrame, const char **name, NSUInteger *line)
JSContext * gOOJSMainThreadContext
static NSMutableDictionary * sObjectConverters
#define OOJS_STACK_SIZE
static BOOL JSNewNSArrayValue(JSContext *context, NSArray *array, jsval *value)
static unsigned sErrorHandlerStackSkip
static BOOL JSNewNSDictionaryValue(JSContext *context, NSDictionary *dictionary, jsval *value)
static void UnregisterObjectConverters(void)
void OOJSInitJSIDCachePRIVATE(const char *name, jsid *idCache)
NSString *const kOOLogException
Definition OOLogging.m:651
BOOL OOLogWillDisplayMessagesInClass(NSString *inMessageClass)
Definition OOLogging.m:144
#define OOLog(class, format,...)
Definition OOLogging.h:88
NSString * OOLogAbbreviatedFileName(const char *inName)
Definition OOLogging.m:839
#define MIN(A, B)
Definition OOMaths.h:111
unsigned count
return nil
@ STELLAR_TYPE_MOON
@ STELLAR_TYPE_ATMOSPHERE
@ STELLAR_TYPE_MINIATURE
@ STELLAR_TYPE_SUN
@ STELLAR_TYPE_NORMAL_PLANET
jsval oo_jsValueInContext:(JSContext *context)
OOJSScript * currentlyRunningScript()
Definition OOJSScript.m:338
OOJavaScriptEngine * sharedEngine()
id jsScriptFromFileNamed:properties:(NSString *fileName,[properties] NSDictionary *properties)
Definition OOScript.m:191
NSDictionary * dictionaryFromFilesNamed:inFolder:andMerge:(NSString *fileName,[inFolder] NSString *folderName,[andMerge] BOOL mergeFiles)