X-Git-Url: https://git.saurik.com/cycript.git/blobdiff_plain/13c7f2fb8fc0619c8da1b5c8a697f7894479b070..2ca4dab349cc3f91032b17ced6b76023b4e67115:/ObjectiveC/Library.mm
diff --git a/ObjectiveC/Library.mm b/ObjectiveC/Library.mm
index a685816..4ef7be8 100644
--- a/ObjectiveC/Library.mm
+++ b/ObjectiveC/Library.mm
@@ -1,38 +1,36 @@
/* Cycript - Optimizing JavaScript Compiler/Runtime
- * Copyright (C) 2009-2013 Jay Freeman (saurik)
+ * Copyright (C) 2009-2014 Jay Freeman (saurik)
*/
-/* GNU General Public License, Version 3 {{{ */
+/* GNU Affero General Public License, Version 3 {{{ */
/*
- * Cycript is free software: you can redistribute it and/or modify
- * it under the terms of the GNU General Public License as published
- * by the Free Software Foundation, either version 3 of the License,
- * or (at your option) any later version.
- *
- * Cycript is distributed in the hope that it will be useful, but
- * WITHOUT ANY WARRANTY; without even the implied warranty of
+ * This program is free software: you can redistribute it and/or modify
+ * it under the terms of the GNU Affero General Public License as published by
+ * the Free Software Foundation, either version 3 of the License, or
+ * (at your option) any later version.
+
+ * This program is distributed in the hope that it will be useful,
+ * but WITHOUT ANY WARRANTY; without even the implied warranty of
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
- * GNU General Public License for more details.
- *
- * You should have received a copy of the GNU General Public License
- * along with Cycript. If not, see .
+ * GNU Affero General Public License for more details.
+
+ * You should have received a copy of the GNU Affero General Public License
+ * along with this program. If not, see .
**/
/* }}} */
-#include
+#include "cycript.hpp"
#include "ObjectiveC/Internal.hpp"
-#include
-
-#include "cycript.hpp"
+#include
+#include
-#include "ObjectiveC/Internal.hpp"
+#include
#ifdef __APPLE__
#include
#include
-#include
#endif
#ifdef __APPLE__
@@ -87,48 +85,9 @@
} return value; \
}
-#ifndef __APPLE__
-#define class_getSuperclass GSObjCSuper
-#define class_getInstanceVariable GSCGetInstanceVariableDefinition
-#define class_getName GSNameFromClass
-
-#define class_removeMethods(cls, list) GSRemoveMethodList(cls, list, YES)
-
-#define ivar_getName(ivar) ((ivar)->ivar_name)
-#define ivar_getOffset(ivar) ((ivar)->ivar_offset)
-#define ivar_getTypeEncoding(ivar) ((ivar)->ivar_type)
-
-#define method_getName(method) ((method)->method_name)
-#define method_getImplementation(method) ((method)->method_imp)
-#define method_getTypeEncoding(method) ((method)->method_types)
-#define method_setImplementation(method, imp) ((void) ((method)->method_imp = (imp)))
-
-#undef objc_getClass
-#define objc_getClass GSClassFromName
-
-#define objc_getProtocol GSProtocolFromName
-
-#define object_getClass GSObjCClass
-
-#define object_getInstanceVariable(object, name, value) ({ \
- objc_ivar *ivar(class_getInstanceVariable(object_getClass(object), name)); \
- _assert(value != NULL); \
- if (ivar != NULL) \
- GSObjCGetVariable(object, ivar_getOffset(ivar), sizeof(void *), value); \
- ivar; \
-})
-
-#define object_setIvar(object, ivar, value) ({ \
- void *data = (value); \
- GSObjCSetVariable(object, ivar_getOffset(ivar), sizeof(void *), &data); \
-})
-
-#define protocol_getName(protocol) [(protocol) name]
-#endif
-
-static void (*$objc_setAssociatedObject)(id object, void *key, id value, objc_AssociationPolicy policy);
-static id (*$objc_getAssociatedObject)(id object, void *key);
-static void (*$objc_removeAssociatedObjects)(id object);
+#define _oassert(test) \
+ if (!(test)) \
+ @throw [NSException exceptionWithName:NSInternalInconsistencyException reason:@"_assert(" #test ")" userInfo:nil];
@class NSBlock;
@@ -200,21 +159,23 @@ const char *CYPoolCString(CYPool &pool, JSContextRef context, NSString *value) {
return string;
}
-JSStringRef CYCopyJSString(JSContextRef context, NSString *value) {
#ifdef __APPLE__
+JSStringRef CYCopyJSString(JSContextRef context, NSString *value) {
return JSStringCreateWithCFString(reinterpret_cast(value));
-#else
- CYPool pool;
- return CYCopyJSString(CYPoolCString(pool, context, value));
-#endif
}
+#endif
JSStringRef CYCopyJSString(JSContextRef context, NSObject *value) {
if (value == nil)
return NULL;
// XXX: this definition scares me; is anyone using this?!
NSString *string([value description]);
+#ifdef __APPLE__
return CYCopyJSString(context, string);
+#else
+ CYPool pool;
+ return CYCopyJSString(CYPoolCString(pool, context, string));
+#endif
}
NSString *CYCopyNSString(const CYUTF8String &value) {
@@ -304,8 +265,6 @@ static Class NSCFBoolean_;
static Class NSCFType_;
static Class NSGenericDeallocHandler_;
static Class NSZombie_;
-
-static std::set banned_;
#else
static Class NSBoolNumber_;
#endif
@@ -370,7 +329,7 @@ JSValueRef CYGetClassPrototype(JSContextRef context, Class self, bool meta) {
prototype = CYGetClassPrototype(context, class_getSuperclass(self), meta);
JSObjectRef object(JSObjectMake(context, _class, NULL));
- JSObjectSetPrototype(context, object, prototype);
+ CYSetPrototype(context, object, prototype);
CYSetProperty(context, cy, name, object);
return object;
@@ -383,7 +342,7 @@ _finline JSValueRef CYGetClassPrototype(JSContextRef context, Class self) {
JSObjectRef Messages::Make(JSContextRef context, Class _class) {
JSObjectRef value(JSObjectMake(context, Messages_, new Messages(_class)));
if (Class super = class_getSuperclass(_class))
- JSObjectSetPrototype(context, value, Messages::Make(context, super));
+ CYSetPrototype(context, value, Messages::Make(context, super));
return value;
}
@@ -406,12 +365,12 @@ bool CYIsKindOfClass(id object, Class _class) {
JSObjectRef Instance::Make(JSContextRef context, id object, Flags flags) {
JSObjectRef value(JSObjectMake(context, CYIsKindOfClass(object, NSBlock_) ? FunctionInstance_ : Instance_, new Instance(object, flags)));
- JSObjectSetPrototype(context, value, CYGetClassPrototype(context, object_getClass(object)));
+ CYSetPrototype(context, value, CYGetClassPrototype(context, object_getClass(object)));
return value;
}
Instance::~Instance() {
- if ((flags_ & Transient) == 0)
+ if ((flags_ & Permanent) == 0)
[GetValue() release];
}
@@ -427,17 +386,28 @@ struct Message_privateData :
}
};
-JSObjectRef CYMakeInstance(JSContextRef context, id object, bool transient) {
- Instance::Flags flags;
+JSObjectRef CYMakeInstance(JSContextRef context, id object, Instance::Flags flags = Instance::None) {
+ _assert(object != nil);
- if (transient)
- flags = Instance::Transient;
- else {
- flags = Instance::None;
+#ifdef __APPLE__
+ JSWeakObjectMapRef weak(CYCastPointer(context, CYGetCachedValue(context, weak_s)));
+
+ if (weak != NULL && &JSWeakObjectMapGet != NULL)
+ if (JSObjectRef instance = JSWeakObjectMapGet(context, weak, object))
+ return instance;
+#endif
+
+ if ((flags & Instance::Permanent) == 0)
object = [object retain];
- }
- return Instance::Make(context, object, flags);
+ JSObjectRef instance(Instance::Make(context, object, flags));
+
+#ifdef __APPLE__
+ if (weak != NULL && &JSWeakObjectMapSet != NULL)
+ JSWeakObjectMapSet(context, weak, object, instance);
+#endif
+
+ return instance;
}
@interface NSMethodSignature (Cycript)
@@ -450,7 +420,7 @@ JSObjectRef CYMakeInstance(JSContextRef context, id object, bool transient) {
- (JSType) cy$JSType;
- (JSValueRef) cy$toJSON:(NSString *)key inContext:(JSContextRef)context;
-- (NSString *) cy$toCYON:(bool)objective;
+- (NSString *) cy$toCYON:(bool)objective inSet:(std::set &)objects;
- (bool) cy$hasProperty:(NSString *)name;
- (NSObject *) cy$getProperty:(NSString *)name;
@@ -468,42 +438,42 @@ JSObjectRef CYMakeInstance(JSContextRef context, id object, bool transient) {
- (JSValueRef) cy$valueOfInContext:(JSContextRef)context;
@end
-NSString *CYCastNSCYON(id value, bool objective) {
- NSString *string;
+NSString *CYCastNSCYON(id value, bool objective, std::set &objects) {
+ _assert(value != nil);
+
+ Class _class(object_getClass(value));
+
+ if (class_isMetaClass(_class)) {
+ const char *name(class_getName(value));
+ if (class_isMetaClass(value))
+ return [NSString stringWithFormat:@"object_getClass(%s)", name];
+ else
+ return [NSString stringWithUTF8String:name];
+ }
- if (value == nil)
- string = @"nil";
- else {
- Class _class(object_getClass(value));
- SEL sel(@selector(cy$toCYON:));
-
- if (objc_method *toCYON = class_getInstanceMethod(_class, sel))
- string = reinterpret_cast(method_getImplementation(toCYON))(value, sel, objective);
- else if (objc_method *methodSignatureForSelector = class_getInstanceMethod(_class, @selector(methodSignatureForSelector:))) {
- if (reinterpret_cast(method_getImplementation(methodSignatureForSelector))(value, @selector(methodSignatureForSelector:), sel) != nil)
- string = [value cy$toCYON:objective];
- else goto fail;
- } else fail: {
- if (false);
#ifdef __APPLE__
- else if (value == NSZombie_)
- string = @"_NSZombie_";
- else if (_class == NSZombie_)
- string = [NSString stringWithFormat:@"<_NSZombie_: %p>", value];
- // XXX: frowny /in/ the pants
- else if (banned_.find(value) != banned_.end())
- string = nil;
+ if (_class == NSZombie_)
+ return [NSString stringWithFormat:@"<_NSZombie_: %p>", value];
#endif
- else
- string = [NSString stringWithFormat:@"%@", value];
- }
- // XXX: frowny pants
- if (string == nil)
- string = @"undefined";
- }
+ SEL sel(@selector(cy$toCYON:inSet:));
- return string;
+ if (objc_method *toCYON = class_getInstanceMethod(_class, sel))
+ return reinterpret_cast &)>(method_getImplementation(toCYON))(value, sel, objective, objects);
+ else if (objc_method *methodSignatureForSelector = class_getInstanceMethod(_class, @selector(methodSignatureForSelector:)))
+ if (reinterpret_cast(method_getImplementation(methodSignatureForSelector))(value, @selector(methodSignatureForSelector:), sel) != nil)
+ return [value cy$toCYON:objective inSet:objects];
+
+ return [NSString stringWithFormat:@"%@", value];
+}
+
+NSString *CYCastNSCYON(id value, bool objective, std::set *objects) {
+ if (objects != NULL)
+ return CYCastNSCYON(value, objective, *objects);
+ else {
+ std::set objects;
+ return CYCastNSCYON(value, objective, objects);
+ }
}
#ifdef __APPLE__
@@ -658,6 +628,7 @@ _finline bool CYJSValueIsInstanceOfCachedConstructor(JSContextRef context, JSVal
return _jsccall(JSValueIsInstanceOfConstructor, context, value, CYGetCachedObject(context, cache));
}
+#ifdef __APPLE__
struct CYBlockDescriptor {
struct {
BlockDescriptor1 one_;
@@ -702,6 +673,7 @@ NSBlock *CYMakeBlock(JSContextRef context, JSObjectRef function, sig::Signature
return reinterpret_cast(literal);
}
+#endif
NSObject *CYCastNSObject(CYPool *pool, JSContextRef context, JSObjectRef object) {
if (CYJSValueIsNSObject(context, object)) {
@@ -792,7 +764,9 @@ NSObject *CYCopyNSObject(CYPool &pool, JSContextRef context, JSValueRef value) {
return [[self mutableCopy] autorelease];
}
-- (NSString *) cy$toCYON:(bool)objective {
+- (NSString *) cy$toCYON:(bool)objective inSet:(std::set &)objects {
+ _oassert(objects.insert(self).second);
+
NSMutableString *json([[[NSMutableString alloc] init] autorelease]);
[json appendString:@"@["];
@@ -807,8 +781,8 @@ NSObject *CYCopyNSObject(CYPool &pool, JSContextRef context, JSValueRef value) {
[json appendString:@","];
else
comma = true;
- if (object == nil || [object cy$JSType] != kJSTypeUndefined)
- [json appendString:CYCastNSCYON(object, true)];
+ if (object != nil && [object cy$JSType] != kJSTypeUndefined)
+ [json appendString:CYCastNSCYON(object, true, objects)];
else {
[json appendString:@","];
comma = false;
@@ -881,7 +855,7 @@ NSObject *CYCopyNSObject(CYPool &pool, JSContextRef context, JSValueRef value) {
return kJSTypeBoolean;
}
-- (NSString *) cy$toCYON:(bool)objective {
+- (NSString *) cy$toCYON:(bool)objective inSet:(std::set &)objects {
NSString *value([self boolValue] ? @"true" : @"false");
return objective ? value : [NSString stringWithFormat:@"@%@", value];
}
@@ -900,7 +874,9 @@ NSObject *CYCopyNSObject(CYPool &pool, JSContextRef context, JSValueRef value) {
return [[self mutableCopy] autorelease];
}
-- (NSString *) cy$toCYON:(bool)objective {
+- (NSString *) cy$toCYON:(bool)objective inSet:(std::set &)objects {
+ _oassert(objects.insert(self).second);
+
NSMutableString *json([[[NSMutableString alloc] init] autorelease]);
[json appendString:@"@{"];
@@ -915,10 +891,10 @@ NSObject *CYCopyNSObject(CYPool &pool, JSContextRef context, JSValueRef value) {
[json appendString:@","];
else
comma = true;
- [json appendString:CYCastNSCYON(key, true)];
+ [json appendString:CYCastNSCYON(key, true, objects)];
[json appendString:@":"];
NSObject *object([self objectForKey:key]);
- [json appendString:CYCastNSCYON(object, true)];
+ [json appendString:CYCastNSCYON(object, true, objects)];
}
[json appendString:@"}"];
@@ -1038,7 +1014,7 @@ NSObject *CYCopyNSObject(CYPool &pool, JSContextRef context, JSValueRef value) {
return kJSTypeNumber;
}
-- (NSString *) cy$toCYON:(bool)objective {
+- (NSString *) cy$toCYON:(bool)objective inSet:(std::set &)objects {
NSString *value([self cy$JSType] != kJSTypeBoolean ? [self stringValue] : [self boolValue] ? @"true" : @"false");
return objective ? value : [NSString stringWithFormat:@"@%@", value];
}
@@ -1056,7 +1032,7 @@ NSObject *CYCopyNSObject(CYPool &pool, JSContextRef context, JSValueRef value) {
return kJSTypeNull;
}
-- (NSString *) cy$toCYON:(bool)objective {
+- (NSString *) cy$toCYON:(bool)objective inSet:(std::set &)objects {
NSString *value(@"null");
return objective ? value : [NSString stringWithFormat:@"@%@", value];
}
@@ -1086,8 +1062,8 @@ NSObject *CYCopyNSObject(CYPool &pool, JSContextRef context, JSValueRef value) {
return kJSTypeObject;
}
-- (NSString *) cy$toCYON:(bool)objective {
- return [@"#" stringByAppendingString:[[self description] cy$toCYON:true]];
+- (NSString *) cy$toCYON:(bool)objective inSet:(std::set &)objects {
+ return [@"#" stringByAppendingString:[[self description] cy$toCYON:true inSet:objects]];
}
- (bool) cy$hasProperty:(NSString *)name {
@@ -1124,8 +1100,8 @@ NSObject *CYCopyNSObject(CYPool &pool, JSContextRef context, JSValueRef value) {
/* Bridge: NSProxy {{{ */
@implementation NSProxy (Cycript)
-- (NSString *) cy$toCYON:(bool)objective {
- return [[self description] cy$toCYON:objective];
+- (NSString *) cy$toCYON:(bool)objective inSet:(std::set &)objects {
+ return [[self description] cy$toCYON:objective inSet:objects];
}
@end
@@ -1133,10 +1109,12 @@ NSObject *CYCopyNSObject(CYPool &pool, JSContextRef context, JSValueRef value) {
/* Bridge: NSSet {{{ */
@implementation NSSet (Cycript)
-- (NSString *) cy$toCYON:(bool)objective {
+- (NSString *) cy$toCYON:(bool)objective inSet:(std::set &)objects {
+ _oassert(objects.insert(self).second);
+
NSMutableString *json([[[NSMutableString alloc] init] autorelease]);
[json appendString:@"[NSSet setWithArray:"];
- [json appendString:CYCastNSCYON([self allObjects], true)];
+ [json appendString:CYCastNSCYON([self allObjects], true, objects)];
[json appendString:@"]]"];
return json;
}
@@ -1154,7 +1132,7 @@ NSObject *CYCopyNSObject(CYPool &pool, JSContextRef context, JSValueRef value) {
return kJSTypeString;
}
-- (NSString *) cy$toCYON:(bool)objective {
+- (NSString *) cy$toCYON:(bool)objective inSet:(std::set &)objects {
std::ostringstream str;
if (!objective)
str << '@';
@@ -1203,7 +1181,7 @@ NSObject *CYCopyNSObject(CYPool &pool, JSContextRef context, JSValueRef value) {
return kJSTypeUndefined;
}
-- (NSString *) cy$toCYON:(bool)objective {
+- (NSString *) cy$toCYON:(bool)objective inSet:(std::set &)objects {
NSString *value(@"undefined");
return value; // XXX: maybe use the below code, adding @undefined?
//return objective ? value : [NSString stringWithFormat:@"@%@", value];
@@ -1245,7 +1223,7 @@ JSValueRef CYCastJSValue(JSContextRef context, NSObject *value) { CYPoolTry {
if (value == nil)
return CYJSNull(context);
else
- return CYMakeInstance(context, value, false);
+ return CYMakeInstance(context, value);
} CYPoolCatch(NULL) return /*XXX*/ NULL; }
@implementation CYJSObject
@@ -1265,11 +1243,11 @@ JSValueRef CYCastJSValue(JSContextRef context, NSObject *value) { CYPoolTry {
[super dealloc];
} CYObjectiveCatch }
-- (NSString *) cy$toCYON:(bool)objective { CYObjectiveTry {
+- (NSString *) cy$toCYON:(bool)objective inSet:(std::set &)objects { CYObjectiveTry {
CYPool pool;
- const char *cyon(CYPoolCCYON(pool, context, object_));
+ const char *cyon(CYPoolCCYON(pool, context, object_, objects));
if (cyon == NULL)
- return [super cy$toCYON:objective];
+ return [super cy$toCYON:objective inSet:objects];
else
return [NSString stringWithUTF8String:cyon];
} CYObjectiveCatch }
@@ -1307,9 +1285,9 @@ JSValueRef CYCastJSValue(JSContextRef context, NSObject *value) { CYPoolTry {
@implementation CYJSArray
-- (NSString *) cy$toCYON:(bool)objective { CYObjectiveTry {
+- (NSString *) cy$toCYON:(bool)objective inSet:(std::set &)objects { CYObjectiveTry {
CYPool pool;
- return [NSString stringWithUTF8String:CYPoolCCYON(pool, context, object_)];
+ return [NSString stringWithUTF8String:CYPoolCCYON(pool, context, object_, objects)];
} CYObjectiveCatch }
- (id) initWithJSObject:(JSObjectRef)object inContext:(JSContextRef)context { CYObjectiveTry_ {
@@ -1429,32 +1407,36 @@ JSValueRef CYCastJSValue(JSContextRef context, NSObject *value) { CYPoolTry {
}
+ (CYInternal *) get:(id)object {
- if ($objc_getAssociatedObject == NULL)
+#ifdef __APPLE__
+ if (&objc_getAssociatedObject == NULL)
return nil;
@synchronized (object) {
- if (CYInternal *internal = $objc_getAssociatedObject(object, @selector(cy$internal)))
+ if (CYInternal *internal = objc_getAssociatedObject(object, @selector(cy$internal)))
return internal;
}
+#endif
return nil;
}
+ (CYInternal *) set:(id)object inContext:(JSContextRef)context {
- if ($objc_getAssociatedObject == NULL)
+#ifdef __APPLE__
+ if (&objc_getAssociatedObject == NULL)
return nil;
@synchronized (object) {
- if (CYInternal *internal = $objc_getAssociatedObject(object, @selector(cy$internal)))
+ if (CYInternal *internal = objc_getAssociatedObject(object, @selector(cy$internal)))
return internal;
- if ($objc_setAssociatedObject == NULL)
+ if (&objc_setAssociatedObject == NULL)
return nil;
CYInternal *internal([[[CYInternal alloc] initInContext:context] autorelease]);
objc_setAssociatedObject(object, @selector(cy$internal), internal, OBJC_ASSOCIATION_RETAIN_NONATOMIC);
return internal;
}
+#endif
return nil;
}
@@ -1484,20 +1466,11 @@ void CYObjectiveC_ExecuteEnd(JSContextRef context, void *handle) { CYSadTry {
return [(NSAutoreleasePool *) handle release];
} CYSadCatch() }
-JSValueRef CYObjectiveC_RuntimeProperty(JSContextRef context, CYUTF8String name) { CYPoolTry {
- if (name == "nil")
- return Instance::Make(context, nil);
- if (Class _class = objc_getClass(name.data))
- return CYMakeInstance(context, _class, true);
- if (Protocol *protocol = objc_getProtocol(name.data))
- return CYMakeInstance(context, protocol, true);
- return NULL;
-} CYPoolCatch(NULL) return /*XXX*/ NULL; }
-
static void CYObjectiveC_CallFunction(JSContextRef context, ffi_cif *cif, void (*function)(), uint8_t *value, void **values) { CYSadTry {
ffi_call(cif, function, value, values);
} CYSadCatch() }
+#ifdef __APPLE__
static NSBlock *CYCastNSBlock(CYPool &pool, JSContextRef context, JSValueRef value, sig::Signature *signature) {
if (JSValueIsNull(context, value))
return nil;
@@ -1532,15 +1505,18 @@ static NSBlock *CYCastNSBlock(CYPool &pool, JSContextRef context, JSValueRef val
return CYMakeBlock(context, object, modified);
}
+#endif
static bool CYObjectiveC_PoolFFI(CYPool *pool, JSContextRef context, sig::Type *type, ffi_type *ffi, void *data, JSValueRef value) { CYSadTry {
// XXX: assigning to an indirect id * works for return values, but not for properties and fields
switch (type->primitive) {
+#ifdef __APPLE__
case sig::block_P:
// XXX: this function might not handle the idea of a null pool
*reinterpret_cast(data) = CYCastNSBlock(*pool, context, value, &type->data.signature);
break;
+#endif
case sig::object_P:
case sig::typename_P:
@@ -1563,19 +1539,32 @@ static JSValueRef CYObjectiveC_FromFFI(JSContextRef context, sig::Type *type, ff
// XXX: do something epic about blocks
case sig::block_P:
case sig::object_P:
- if (NSObject *object = *reinterpret_cast(data)) {
- JSValueRef value(CYCastJSValue(context, object));
- if (initialize)
- [object release];
- return value;
+ if (NSObject *value = *reinterpret_cast(data)) {
+ JSObjectRef object(CYMakeInstance(context, value));
+
+ if (initialize) {
+ Instance *internal(reinterpret_cast(JSObjectGetPrivate(object)));
+
+ if ((internal->flags_ & Instance::Uninitialized) != 0) {
+ internal->flags_ = static_cast(internal->flags_ & ~Instance::Uninitialized);
+ _assert(internal->value_ == nil);
+ internal->value_ = value;
+ }
+
+ [value release];
+ }
+
+ return object;
} else goto null;
case sig::typename_P:
- return CYMakeInstance(context, *reinterpret_cast(data), true);
+ if (Class value = *reinterpret_cast(data))
+ return CYMakeInstance(context, value, Instance::Permanent);
+ else goto null;
case sig::selector_P:
- if (SEL sel = *reinterpret_cast(data))
- return CYMakeSelector(context, sel);
+ if (SEL value = *reinterpret_cast(data))
+ return CYMakeSelector(context, value);
else goto null;
null:
@@ -1676,7 +1665,6 @@ static bool Messages_setProperty(JSContextRef context, JSObjectRef object, JSStr
} else _assert(false);
objc_method *method(NULL);
-#if OBJC_API_VERSION >= 2
unsigned int size;
objc_method **methods(class_copyMethodList(_class, &size));
for (size_t i(0); i != size; ++i)
@@ -1685,65 +1673,24 @@ static bool Messages_setProperty(JSContextRef context, JSObjectRef object, JSStr
break;
}
free(methods);
-#else
- for (objc_method_list *methods(_class->methods); methods != NULL; methods = methods->method_next)
- for (int i(0); i != methods->method_count; ++i)
- if (sel_isEqual(method_getName(&methods->method_list[i]), sel)) {
- method = &methods->method_list[i];
- break;
- }
-#endif
if (method != NULL)
method_setImplementation(method, imp);
- else {
-#ifdef GNU_RUNTIME
- GSMethodList list(GSAllocMethodList(1));
- GSAppendMethodToList(list, sel, type, imp, YES);
- GSAddMethodList(_class, list, YES);
- GSFlushMethodCacheForClass(_class);
-#else
+ else
class_addMethod(_class, sel, imp, type);
-#endif
- }
return true;
} CYCatch(false) }
-#if 0 && OBJC_API_VERSION < 2
-static bool Messages_deleteProperty(JSContextRef context, JSObjectRef object, JSStringRef property, JSValueRef *exception) { CYTry {
- Messages *internal(reinterpret_cast(JSObjectGetPrivate(object)));
- Class _class(internal->GetValue());
-
- CYPool pool;
- const char *name(CYPoolCString(pool, context, property));
-
- if (SEL sel = sel_getUid(name))
- if (objc_method *method = class_getInstanceMethod(_class, sel)) {
- objc_method_list list = {NULL, 1, {method}};
- class_removeMethods(_class, &list);
- return true;
- }
-
- return false;
-} CYCatch(false) }
-#endif
-
static void Messages_getPropertyNames(JSContextRef context, JSObjectRef object, JSPropertyNameAccumulatorRef names) {
Messages *internal(reinterpret_cast(JSObjectGetPrivate(object)));
Class _class(internal->GetValue());
-#if OBJC_API_VERSION >= 2
unsigned int size;
objc_method **data(class_copyMethodList(_class, &size));
for (size_t i(0); i != size; ++i)
JSPropertyNameAccumulatorAddName(names, CYJSString(sel_getName(method_getName(data[i]))));
free(data);
-#else
- for (objc_method_list *methods(_class->methods); methods != NULL; methods = methods->method_next)
- for (int i(0); i != methods->method_count; ++i)
- JSPropertyNameAccumulatorAddName(names, CYJSString(sel_getName(method_getName(&methods->method_list[i]))));
-#endif
}
static bool CYHasImplicitProperties(Class _class) {
@@ -1930,17 +1877,11 @@ static void Instance_getPropertyNames(JSContextRef context, JSObjectRef object,
if (CYHasImplicitProperties(_class))
for (Class current(_class); current != nil; current = class_getSuperclass(current)) {
-#if OBJC_API_VERSION >= 2
unsigned int size;
objc_method **data(class_copyMethodList(current, &size));
for (size_t i(0); i != size; ++i)
Instance_getPropertyNames_message(names, data[i]);
free(data);
-#else
- for (objc_method_list *methods(current->methods); methods != NULL; methods = methods->method_next)
- for (int i(0); i != methods->method_count; ++i)
- Instance_getPropertyNames_message(names, &methods->method_list[i]);
-#endif
}
CYPoolTry {
@@ -1952,7 +1893,7 @@ static void Instance_getPropertyNames(JSContextRef context, JSObjectRef object,
static JSObjectRef Instance_callAsConstructor(JSContextRef context, JSObjectRef object, size_t count, const JSValueRef arguments[], JSValueRef *exception) { CYTry {
Instance *internal(reinterpret_cast(JSObjectGetPrivate(object)));
- JSObjectRef value(Instance::Make(context, [internal->GetValue() alloc], Instance::Uninitialized));
+ JSObjectRef value(CYMakeInstance(context, [internal->GetValue() alloc], Instance::Uninitialized));
return value;
} CYCatch(NULL) }
@@ -2048,6 +1989,7 @@ static void CYBitField(unsigned &length, unsigned &shift, id self, Ivar ivar, co
break;
else if (ivar_getOffset(ivars[i]) == offset) {
const char *encoding(ivar_getTypeEncoding(ivars[i]));
+ _assert(encoding != NULL);
_assert(encoding[0] == 'b');
shift += CYCastDouble(encoding + 1);
}
@@ -2071,6 +2013,8 @@ static JSValueRef Internal_getProperty(JSContextRef context, JSObjectRef object,
void *data(reinterpret_cast(self) + offset);
const char *encoding(ivar_getTypeEncoding(ivar));
+ _assert(encoding != NULL);
+ _assert(encoding[0] != '\0');
if (encoding[0] == 'b') {
unsigned length, shift;
CYBitField(length, shift, self, ivar, encoding, offset);
@@ -2079,8 +2023,8 @@ static JSValueRef Internal_getProperty(JSContextRef context, JSObjectRef object,
uintptr_t mask((1 << length) - 1);
return CYCastJSValue(context, (field >> shift) & mask);
} else {
- Type_privateData type(pool, ivar_getTypeEncoding(ivar));
- return CYFromFFI(context, type.type_, type.GetFFI(), data);
+ auto type(new(pool) Type_privateData(encoding));
+ return CYFromFFI(context, type->type_, type->GetFFI(), data);
}
}
@@ -2099,6 +2043,7 @@ static bool Internal_setProperty(JSContextRef context, JSObjectRef object, JSStr
void *data(reinterpret_cast(self) + offset);
const char *encoding(ivar_getTypeEncoding(ivar));
+ _assert(encoding != NULL);
if (encoding[0] == 'b') {
unsigned length, shift;
CYBitField(length, shift, self, ivar, encoding, offset);
@@ -2107,8 +2052,8 @@ static bool Internal_setProperty(JSContextRef context, JSObjectRef object, JSStr
uintptr_t mask((1 << length) - 1);
field = field & ~(mask << shift) | (uintptr_t(CYCastDouble(context, value)) & mask) << shift;
} else {
- Type_privateData type(pool, ivar_getTypeEncoding(ivar));
- CYPoolFFI(&pool, context, type.type_, type.GetFFI(), reinterpret_cast(self) + ivar_getOffset(ivar), value);
+ auto type(new(pool) Type_privateData(ivar_getTypeEncoding(ivar)));
+ CYPoolFFI(&pool, context, type->type_, type->GetFFI(), reinterpret_cast(self) + ivar_getOffset(ivar), value);
return true;
}
}
@@ -2120,17 +2065,11 @@ static void Internal_getPropertyNames_(Class _class, JSPropertyNameAccumulatorRe
if (Class super = class_getSuperclass(_class))
Internal_getPropertyNames_(super, names);
-#if OBJC_API_VERSION >= 2
unsigned int size;
objc_ivar **data(class_copyIvarList(_class, &size));
for (size_t i(0); i != size; ++i)
JSPropertyNameAccumulatorAddName(names, CYJSString(ivar_getName(data[i])));
free(data);
-#else
- if (objc_ivar_list *ivars = _class->ivars)
- for (int i(0); i != ivars->ivar_count; ++i)
- JSPropertyNameAccumulatorAddName(names, CYJSString(ivar_getName(&ivars->ivar_list[i])));
-#endif
}
static void Internal_getPropertyNames(JSContextRef context, JSObjectRef object, JSPropertyNameAccumulatorRef names) {
@@ -2157,11 +2096,10 @@ static JSValueRef ObjectiveC_Classes_getProperty(JSContextRef context, JSObjectR
CYPool pool;
NSString *name(CYCastNSString(&pool, context, property));
if (Class _class = NSClassFromString(name))
- return CYMakeInstance(context, _class, true);
+ return CYMakeInstance(context, _class, Instance::Permanent);
return NULL;
} CYCatch(NULL) }
-#ifdef __APPLE__
static Class *CYCopyClassList(size_t &size) {
size = objc_getClassList(NULL, 0);
Class *data(reinterpret_cast(malloc(sizeof(Class) * size)));
@@ -2183,21 +2121,14 @@ static Class *CYCopyClassList(size_t &size) {
size = writ;
}
}
-#endif
static void ObjectiveC_Classes_getPropertyNames(JSContextRef context, JSObjectRef object, JSPropertyNameAccumulatorRef names) {
-#ifdef __APPLE__
size_t size;
if (Class *data = CYCopyClassList(size)) {
for (size_t i(0); i != size; ++i)
JSPropertyNameAccumulatorAddName(names, CYJSString(class_getName(data[i])));
free(data);
}
-#else
- void *state(NULL);
- while (Class _class = objc_next_class(&state))
- JSPropertyNameAccumulatorAddName(names, CYJSString(class_getName(_class)));
-#endif
}
#if OBJC_API_VERSION >= 2
@@ -2212,7 +2143,7 @@ static JSValueRef ObjectiveC_Image_Classes_getProperty(JSContextRef context, JSO
for (size_t i(0); i != size; ++i)
if (strcmp(name, data[i]) == 0) {
if (Class _class = objc_getClass(name)) {
- value = CYMakeInstance(context, _class, true);
+ value = CYMakeInstance(context, _class, Instance::Permanent);
goto free;
} else
break;
@@ -2265,7 +2196,7 @@ static JSValueRef ObjectiveC_Protocols_getProperty(JSContextRef context, JSObjec
CYPool pool;
const char *name(CYPoolCString(pool, context, property));
if (Protocol *protocol = objc_getProtocol(name))
- return CYMakeInstance(context, protocol, true);
+ return CYMakeInstance(context, protocol, Instance::Permanent);
return NULL;
} CYCatch(NULL) }
@@ -2285,7 +2216,7 @@ static JSValueRef ObjectiveC_Constants_getProperty(JSContextRef context, JSObjec
CYPool pool;
CYUTF8String name(CYPoolUTF8String(pool, context, property));
if (name == "nil")
- return Instance::Make(context, nil);
+ return CYJSNull(context);
return NULL;
} CYCatch(NULL) }
@@ -2571,7 +2502,7 @@ static JSObjectRef Instance_new(JSContextRef context, JSObjectRef object, size_t
if (count > 1)
throw CYJSError(context, "incorrect number of arguments to Instance constructor");
id self(count == 0 ? nil : CYCastPointer(context, arguments[0]));
- return CYMakeInstance(context, self, false);
+ return CYMakeInstance(context, self);
} CYCatch(NULL) }
static JSValueRef CYValue_getProperty_value(JSContextRef context, JSObjectRef object, JSStringRef property, JSValueRef *exception) { CYTry {
@@ -2611,7 +2542,7 @@ static JSValueRef FunctionInstance_getProperty_type(JSContextRef context, JSObje
static JSValueRef Instance_getProperty_constructor(JSContextRef context, JSObjectRef object, JSStringRef property, JSValueRef *exception) { CYTry {
Instance *internal(reinterpret_cast(JSObjectGetPrivate(object)));
- return Instance::Make(context, (id) object_getClass(internal->GetValue()));
+ return CYMakeInstance(context, object_getClass(internal->GetValue()), Instance::Permanent);
} CYCatch(NULL) }
static JSValueRef Instance_getProperty_prototype(JSContextRef context, JSObjectRef object, JSStringRef property, JSValueRef *exception) { CYTry {
@@ -2631,11 +2562,13 @@ static JSValueRef Instance_getProperty_messages(JSContextRef context, JSObjectRe
} CYCatch(NULL) }
static JSValueRef Instance_callAsFunction_toCYON(JSContextRef context, JSObjectRef object, JSObjectRef _this, size_t count, const JSValueRef arguments[], JSValueRef *exception) { CYTry {
+ std::set *objects(CYCastObjects(context, _this, count, arguments));
+
if (!CYJSValueIsNSObject(context, _this))
return NULL;
Instance *internal(reinterpret_cast(JSObjectGetPrivate(_this)));
- return CYCastJSValue(context, CYJSString(context, CYCastNSCYON(internal->GetValue(), false)));
+ return CYCastJSValue(context, CYJSString(context, CYCastNSCYON(internal->GetValue(), false, objects)));
} CYCatch(NULL) }
static JSValueRef Instance_callAsFunction_toJSON(JSContextRef context, JSObjectRef object, JSObjectRef _this, size_t count, const JSValueRef arguments[], JSValueRef *exception) { CYTry {
@@ -2667,6 +2600,7 @@ static JSValueRef Instance_callAsFunction_valueOf(JSContextRef context, JSObject
Instance *internal(reinterpret_cast(JSObjectGetPrivate(_this)));
id value(internal->GetValue());
+ _assert(value != nil);
if (![value respondsToSelector:@selector(cy$valueOfInContext:)])
return _this;
@@ -2709,16 +2643,11 @@ static JSValueRef Class_callAsFunction_pointerTo(JSContextRef context, JSObjectR
if (!CYIsClass(value))
CYThrow("non-Class object cannot be used as Type");
- // XXX: this is a very silly implementation
-
- std::ostringstream type;
- type << "@\"";
- type << class_getName(value);
- type << "\"";
-
- CYPoolTry {
- return CYMakeType(context, type.str().c_str());
- } CYPoolCatch(NULL)
+ sig::Type type;
+ memset(&type, 0, sizeof(type));
+ type.primitive = sig::object_P;
+ type.name = class_getName(value);
+ return CYMakeType(context, &type);
} CYCatch(NULL) return /*XXX*/ NULL; }
static JSValueRef Selector_callAsFunction_toString(JSContextRef context, JSObjectRef object, JSObjectRef _this, size_t count, const JSValueRef arguments[], JSValueRef *exception) { CYTry {
@@ -2816,14 +2745,10 @@ JSValueRef NSCFType$cy$toJSON$inContext$(id self, SEL sel, JSValueRef key, JSCon
#endif
void CYObjectiveC_Initialize() { /*XXX*/ JSContextRef context(NULL); CYPoolTry {
- $objc_setAssociatedObject = reinterpret_cast(dlsym(RTLD_DEFAULT, "objc_setAssociatedObject"));
- $objc_getAssociatedObject = reinterpret_cast(dlsym(RTLD_DEFAULT, "objc_getAssociatedObject"));
- $objc_removeAssociatedObjects = reinterpret_cast(dlsym(RTLD_DEFAULT, "objc_removeAssociatedObjects"));
-
CYPool &pool(CYGetGlobalPool());
- Object_type = new(pool) Type_privateData("@");
- Selector_type = new(pool) Type_privateData(":");
+ Object_type = new(pool) Type_privateData(sig::object_P);
+ Selector_type = new(pool) Type_privateData(sig::selector_P);
NSArray_ = objc_getClass("NSArray");
NSBlock_ = objc_getClass("NSBlock");
@@ -2843,12 +2768,6 @@ void CYObjectiveC_Initialize() { /*XXX*/ JSContextRef context(NULL); CYPoolTry {
NSCFType_ = objc_getClass("NSCFType");
NSZombie_ = objc_getClass("_NSZombie_");
-
- banned_.insert(Object_);
- banned_.insert(objc_getClass("__NSAtom"));
- banned_.insert(objc_getClass("__NSGenericDeallocHandler"));
- banned_.insert(objc_getClass("NSMessageBuilder"));
- banned_.insert(objc_getClass("__NSMessageBuilder"));
#else
NSBoolNumber_ = objc_getClass("NSBoolNumber");
#endif
@@ -2917,9 +2836,6 @@ void CYObjectiveC_Initialize() { /*XXX*/ JSContextRef context(NULL); CYPoolTry {
definition.hasProperty = &Messages_hasProperty;
definition.getProperty = &Messages_getProperty;
definition.setProperty = &Messages_setProperty;
-#if 0 && OBJC_API_VERSION < 2
- definition.deleteProperty = &Messages_deleteProperty;
-#endif
definition.getPropertyNames = &Messages_getPropertyNames;
definition.finalize = &CYFinalize;
Messages_ = JSClassCreate(&definition);
@@ -2972,12 +2888,14 @@ void CYObjectiveC_Initialize() { /*XXX*/ JSContextRef context(NULL); CYPoolTry {
ObjectiveC_Protocols_ = JSClassCreate(&definition);
#ifdef __APPLE__
-// XXX: this is horrible; there has to be a better way to do this
-#ifdef __LP64__
- class_addMethod(NSCFType_, @selector(cy$toJSON:inContext:), reinterpret_cast(&NSCFType$cy$toJSON$inContext$), "^{OpaqueJSValue=}32@0:8@16^{OpaqueJSContext=}24");
-#else
- class_addMethod(NSCFType_, @selector(cy$toJSON:inContext:), reinterpret_cast(&NSCFType$cy$toJSON$inContext$), "^{OpaqueJSValue=}16@0:4@8^{OpaqueJSContext=}12");
-#endif
+ class_addMethod(NSCFType_, @selector(cy$toJSON:inContext:), reinterpret_cast(&NSCFType$cy$toJSON$inContext$),
+ // XXX: this is horrible; there has to be a better way to do this
+ #ifdef __LP64__
+ "^{OpaqueJSValue=}32@0:8@16^{OpaqueJSContext=}24"
+ #else
+ "^{OpaqueJSValue=}16@0:4@8^{OpaqueJSContext=}12"
+ #endif
+ );
#endif
} CYPoolCatch() }
@@ -3020,41 +2938,41 @@ void CYObjectiveC_SetupContext(JSContextRef context) { CYPoolTry {
JSObjectRef ArrayInstance_prototype(CYCastJSObject(context, CYGetProperty(context, ArrayInstance, prototype_s)));
CYSetProperty(context, cy, CYJSString("ArrayInstance_prototype"), ArrayInstance_prototype);
JSObjectRef Array_prototype(CYGetCachedObject(context, CYJSString("Array_prototype")));
- JSObjectSetPrototype(context, ArrayInstance_prototype, Array_prototype);
+ CYSetPrototype(context, ArrayInstance_prototype, Array_prototype);
JSObjectRef BooleanInstance(JSObjectMakeConstructor(context, BooleanInstance_, NULL));
JSObjectRef BooleanInstance_prototype(CYCastJSObject(context, CYGetProperty(context, BooleanInstance, prototype_s)));
CYSetProperty(context, cy, CYJSString("BooleanInstance_prototype"), BooleanInstance_prototype);
JSObjectRef Boolean_prototype(CYGetCachedObject(context, CYJSString("Boolean_prototype")));
- JSObjectSetPrototype(context, BooleanInstance_prototype, Boolean_prototype);
+ CYSetPrototype(context, BooleanInstance_prototype, Boolean_prototype);
JSObjectRef FunctionInstance(JSObjectMakeConstructor(context, FunctionInstance_, NULL));
JSObjectRef FunctionInstance_prototype(CYCastJSObject(context, CYGetProperty(context, FunctionInstance, prototype_s)));
CYSetProperty(context, cy, CYJSString("FunctionInstance_prototype"), FunctionInstance_prototype);
JSObjectRef Function_prototype(CYGetCachedObject(context, CYJSString("Function_prototype")));
- JSObjectSetPrototype(context, FunctionInstance_prototype, Function_prototype);
+ CYSetPrototype(context, FunctionInstance_prototype, Function_prototype);
JSObjectRef NumberInstance(JSObjectMakeConstructor(context, NumberInstance_, NULL));
JSObjectRef NumberInstance_prototype(CYCastJSObject(context, CYGetProperty(context, NumberInstance, prototype_s)));
CYSetProperty(context, cy, CYJSString("NumberInstance_prototype"), NumberInstance_prototype);
JSObjectRef Number_prototype(CYGetCachedObject(context, CYJSString("Number_prototype")));
- JSObjectSetPrototype(context, NumberInstance_prototype, Number_prototype);
+ CYSetPrototype(context, NumberInstance_prototype, Number_prototype);
JSObjectRef ObjectInstance(JSObjectMakeConstructor(context, ObjectInstance_, NULL));
JSObjectRef ObjectInstance_prototype(CYCastJSObject(context, CYGetProperty(context, ObjectInstance, prototype_s)));
CYSetProperty(context, cy, CYJSString("ObjectInstance_prototype"), ObjectInstance_prototype);
JSObjectRef Object_prototype(CYGetCachedObject(context, CYJSString("Object_prototype")));
- JSObjectSetPrototype(context, ObjectInstance_prototype, Object_prototype);
+ CYSetPrototype(context, ObjectInstance_prototype, Object_prototype);
JSObjectRef StringInstance(JSObjectMakeConstructor(context, StringInstance_, NULL));
JSObjectRef StringInstance_prototype(CYCastJSObject(context, CYGetProperty(context, StringInstance, prototype_s)));
CYSetProperty(context, cy, CYJSString("StringInstance_prototype"), StringInstance_prototype);
JSObjectRef String_prototype(CYGetCachedObject(context, CYJSString("String_prototype")));
- JSObjectSetPrototype(context, StringInstance_prototype, String_prototype);
+ CYSetPrototype(context, StringInstance_prototype, String_prototype);
JSObjectRef Class_prototype(CYCastJSObject(context, CYGetProperty(context, Class, prototype_s)));
CYSetProperty(context, cy, CYJSString("Class_prototype"), Class_prototype);
- JSObjectSetPrototype(context, Class_prototype, Instance_prototype);
+ CYSetPrototype(context, Class_prototype, Instance_prototype);
CYSetProperty(context, cycript, CYJSString("Instance"), Instance);
CYSetProperty(context, cycript, CYJSString("Selector"), Selector);
@@ -3069,11 +2987,11 @@ void CYObjectiveC_SetupContext(JSContextRef context) { CYPoolTry {
CYSetProperty(context, all, CYJSString("objc_msgSend"), &$objc_msgSend, kJSPropertyAttributeDontEnum);
- JSObjectSetPrototype(context, CYCastJSObject(context, CYGetProperty(context, Message, prototype_s)), Function_prototype);
- JSObjectSetPrototype(context, CYCastJSObject(context, CYGetProperty(context, Selector, prototype_s)), Function_prototype);
+ CYSetPrototype(context, CYCastJSObject(context, CYGetProperty(context, Message, prototype_s)), Function_prototype);
+ CYSetPrototype(context, CYCastJSObject(context, CYGetProperty(context, Selector, prototype_s)), Function_prototype);
} CYPoolCatch() }
-static CYHooks CYObjectiveCHooks = {
+static CYHook CYObjectiveCHook = {
&CYObjectiveC_ExecuteStart,
&CYObjectiveC_ExecuteEnd,
&CYObjectiveC_CallFunction,
@@ -3083,13 +3001,7 @@ static CYHooks CYObjectiveCHooks = {
&CYObjectiveC_FromFFI,
};
-struct CYObjectiveC {
- CYObjectiveC() {
- hooks_ = &CYObjectiveCHooks;
- // XXX: evil magic juju to make this actually take effect on a Mac when compiled with autoconf/libtool doom!
- _assert(hooks_ != NULL);
- }
-} CYObjectiveC;
+CYRegisterHook CYObjectiveC(&CYObjectiveCHook);
extern "C" void CydgetSetupContext(JSGlobalContextRef context) { CYObjectiveTry_ {
CYSetupContext(context);