X-Git-Url: https://git.saurik.com/cycript.git/blobdiff_plain/179d115012d6dc6a897e1a6ccac70248c9e65081..a703494adeea8b4b225e7bb58eacf94e7f881d01:/ObjectiveC/Library.mm
diff --git a/ObjectiveC/Library.mm b/ObjectiveC/Library.mm
index 52df126..dccc3ad 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-2015 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__
@@ -91,49 +89,6 @@
if (!(test)) \
@throw [NSException exceptionWithName:NSInternalInconsistencyException reason:@"_assert(" #test ")" userInfo:nil];
-#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);
-
@class NSBlock;
struct BlockLiteral {
@@ -204,21 +159,23 @@ const char *CYPoolCString(CYPool &pool, JSContextRef context, NSString *value) {
return string;
}
+#ifdef __clang__
JSStringRef CYCopyJSString(JSContextRef context, NSString *value) {
-#ifdef __APPLE__
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 __clang__
return CYCopyJSString(context, string);
+#else
+ CYPool pool;
+ return CYCopyJSString(CYPoolCString(pool, context, string));
+#endif
}
NSString *CYCopyNSString(const CYUTF8String &value) {
@@ -307,7 +264,6 @@ static Class __NSMallocBlock__;
static Class NSCFBoolean_;
static Class NSCFType_;
static Class NSGenericDeallocHandler_;
-static Class NSZombie_;
#else
static Class NSBoolNumber_;
#endif
@@ -317,6 +273,7 @@ static Class NSBlock_;
static Class NSDictionary_;
static Class NSNumber_;
static Class NSString_;
+static Class NSZombie_;
static Class Object_;
static Type_privateData *Object_type;
@@ -413,7 +370,7 @@ JSObjectRef Instance::Make(JSContextRef context, id object, Flags flags) {
}
Instance::~Instance() {
- if ((flags_ & Transient) == 0)
+ if ((flags_ & Permanent) == 0)
[GetValue() release];
}
@@ -429,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)
@@ -471,42 +439,30 @@ JSObjectRef CYMakeInstance(JSContextRef context, id object, bool transient) {
@end
NSString *CYCastNSCYON(id value, bool objective, std::set &objects) {
- NSString *string;
+ _assert(value != nil);
- if (value == nil)
- string = @"nil";
- else {
- Class _class(object_getClass(value));
- SEL sel(@selector(cy$toCYON:inSet:));
+ Class _class(object_getClass(value));
- if (class_isMetaClass(_class)) {
- const char *name(class_getName(value));
- if (class_isMetaClass(value))
- string = [NSString stringWithFormat:@"object_getClass(%s)", name];
- else
- string = [NSString stringWithUTF8String:name];
- } else if (objc_method *toCYON = class_getInstanceMethod(_class, sel))
- string = 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)
- string = [value cy$toCYON:objective inSet:objects];
- else goto fail;
- } else fail: {
- if (false);
-#ifdef __APPLE__
- else if (_class == NSZombie_)
- string = [NSString stringWithFormat:@"<_NSZombie_: %p>", value];
-#endif
- else
- string = [NSString stringWithFormat:@"%@", value];
- }
-
- // XXX: frowny pants
- if (string == nil)
- string = @"undefined";
+ 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];
}
- return string;
+ if (_class == NSZombie_)
+ return [NSString stringWithFormat:@"<_NSZombie_: %p>", value];
+
+ SEL sel(@selector(cy$toCYON:inSet:));
+
+ 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) {
@@ -518,7 +474,6 @@ NSString *CYCastNSCYON(id value, bool objective, std::set *objects) {
}
}
-#ifdef __APPLE__
struct PropertyAttributes {
CYPool pool_;
@@ -606,7 +561,6 @@ struct PropertyAttributes {
}
};
-#endif
@interface CYWebUndefined : NSObject {
}
@@ -670,6 +624,7 @@ _finline bool CYJSValueIsInstanceOfCachedConstructor(JSContextRef context, JSVal
return _jsccall(JSValueIsInstanceOfConstructor, context, value, CYGetCachedObject(context, cache));
}
+#ifdef __APPLE__
struct CYBlockDescriptor {
struct {
BlockDescriptor1 one_;
@@ -689,10 +644,6 @@ static JSValueRef BlockAdapter_(JSContextRef context, size_t count, JSValueRef v
return CYCallAsFunction(context, function, _this, count - 1, values + 1);
}
-static void BlockClosure_(ffi_cif *cif, void *result, void **arguments, void *arg) {
- CYExecuteClosure(cif, result, arguments, arg, &BlockAdapter_);
-}
-
NSBlock *CYMakeBlock(JSContextRef context, JSObjectRef function, sig::Signature &signature) {
_assert(__NSMallocBlock__ != Nil);
BlockLiteral *literal(reinterpret_cast(malloc(sizeof(BlockLiteral))));
@@ -700,7 +651,7 @@ NSBlock *CYMakeBlock(JSContextRef context, JSObjectRef function, sig::Signature
CYBlockDescriptor *descriptor(new CYBlockDescriptor);
memset(&descriptor->d_, 0, sizeof(descriptor->d_));
- descriptor->internal_ = CYMakeFunctor_(context, function, signature, &BlockClosure_);
+ descriptor->internal_ = CYMakeFunctor_(context, function, signature, &BlockAdapter_);
literal->invoke = reinterpret_cast(descriptor->internal_->GetValue());
literal->isa = __NSMallocBlock__;
@@ -714,6 +665,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)) {
@@ -811,7 +763,7 @@ NSObject *CYCopyNSObject(CYPool &pool, JSContextRef context, JSValueRef value) {
[json appendString:@"@["];
bool comma(false);
-#ifdef __APPLE__
+#ifdef __clang__
for (id object in self) {
#else
for (size_t index(0), count([self count]); index != count; ++index) {
@@ -821,7 +773,7 @@ NSObject *CYCopyNSObject(CYPool &pool, JSContextRef context, JSValueRef value) {
[json appendString:@","];
else
comma = true;
- if (object == nil || [object cy$JSType] != kJSTypeUndefined)
+ if (object != nil && [object cy$JSType] != kJSTypeUndefined)
[json appendString:CYCastNSCYON(object, true, objects)];
else {
[json appendString:@","];
@@ -921,7 +873,7 @@ NSObject *CYCopyNSObject(CYPool &pool, JSContextRef context, JSValueRef value) {
[json appendString:@"@{"];
bool comma(false);
-#ifdef __APPLE__
+#ifdef __clang__
for (NSObject *key in self) {
#else
NSEnumerator *keys([self keyEnumerator]);
@@ -952,7 +904,7 @@ NSObject *CYCopyNSObject(CYPool &pool, JSContextRef context, JSValueRef value) {
- (void) cy$getPropertyNames:(JSPropertyNameAccumulatorRef)names inContext:(JSContextRef)context {
[super cy$getPropertyNames:names inContext:context];
-#ifdef __APPLE__
+#ifdef __clang__
for (NSObject *key in self) {
#else
NSEnumerator *keys([self keyEnumerator]);
@@ -975,11 +927,7 @@ NSObject *CYCopyNSObject(CYPool &pool, JSContextRef context, JSValueRef value) {
if ([name isEqualToString:@"length"]) {
// XXX: is this not intelligent?
NSNumber *number(reinterpret_cast(value));
-#ifdef __APPLE__
NSUInteger size([number unsignedIntegerValue]);
-#else
- NSUInteger size([number unsignedIntValue]);
-#endif
NSUInteger count([self count]);
if (size < count)
[self removeObjectsInRange:NSMakeRange(size, count - size)];
@@ -1135,6 +1083,21 @@ NSObject *CYCopyNSObject(CYPool &pool, JSContextRef context, JSValueRef value) {
return true;
}
+@end
+/* }}} */
+/* Bridge: NSOrderedSet {{{ */
+@implementation NSOrderedSet (Cycript)
+
+- (NSString *) cy$toCYON:(bool)objective inSet:(std::set &)objects {
+ _oassert(objects.insert(self).second);
+
+ NSMutableString *json([[[NSMutableString alloc] init] autorelease]);
+ [json appendString:@"[NSOrderedSet orderedSetWithArray:"];
+ [json appendString:CYCastNSCYON([self array], true, objects)];
+ [json appendString:@"]]"];
+ return json;
+}
+
@end
/* }}} */
/* Bridge: NSProxy {{{ */
@@ -1235,11 +1198,7 @@ NSObject *CYCopyNSObject(CYPool &pool, JSContextRef context, JSValueRef value) {
/* }}} */
static bool CYIsClass(id self) {
-#ifdef __APPLE__
return class_isMetaClass(object_getClass(self));
-#else
- return GSObjCIsClass(self);
-#endif
}
Class CYCastClass(CYPool &pool, JSContextRef context, JSValueRef value) {
@@ -1263,7 +1222,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
@@ -1447,32 +1406,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;
}
@@ -1502,20 +1465,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);
+static void CYObjectiveC_CallFunction(CYPool &pool, JSContextRef context, ffi_cif *cif, void (*function)(), void *value, void **values) { CYSadTry {
+ CYCallFunction(pool, context, 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;
@@ -1550,15 +1504,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:
@@ -1581,19 +1538,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:
@@ -1607,12 +1577,8 @@ static bool CYImplements(id object, Class _class, SEL selector, bool devoid = fa
if (objc_method *method = class_getInstanceMethod(_class, selector)) {
if (!devoid)
return true;
-#if OBJC_API_VERSION >= 2
char type[16];
method_getReturnType(method, type, sizeof(type));
-#else
- const char *type(method_getTypeEncoding(method));
-#endif
if (type[0] != 'v')
return true;
}
@@ -1626,10 +1592,6 @@ static JSValueRef MessageAdapter_(JSContextRef context, size_t count, JSValueRef
return CYCallAsFunction(context, function, _this, count - 2, values + 2);
}
-static void MessageClosure_(ffi_cif *cif, void *result, void **arguments, void *arg) {
- CYExecuteClosure(cif, result, arguments, arg, &MessageAdapter_);
-}
-
static JSObjectRef CYMakeMessage(JSContextRef context, SEL sel, IMP imp, const char *type) {
Message_privateData *internal(new Message_privateData(sel, type, imp));
return JSObjectMake(context, Message_, internal);
@@ -1640,7 +1602,7 @@ static IMP CYMakeMessage(JSContextRef context, JSValueRef value, const char *enc
CYPool pool;
sig::Signature signature;
sig::Parse(pool, &signature, encoding, &Structor_);
- Closure_privateData *internal(CYMakeFunctor_(context, function, signature, &MessageClosure_));
+ Closure_privateData *internal(CYMakeFunctor_(context, function, signature, &MessageAdapter_));
// XXX: see notes in Library.cpp about needing to leak
return reinterpret_cast(internal->GetValue());
}
@@ -1694,7 +1656,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)
@@ -1703,65 +1664,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) {
@@ -1796,10 +1716,8 @@ static bool Instance_hasProperty(JSContextRef context, JSObjectRef object, JSStr
const char *string(CYPoolCString(pool, context, name));
-#ifdef __APPLE__
if (class_getProperty(_class, string) != NULL)
return true;
-#endif
if (CYHasImplicitProperties(_class))
if (SEL sel = sel_getUid(string))
@@ -1831,13 +1749,11 @@ static JSValueRef Instance_getProperty(JSContextRef context, JSObjectRef object,
const char *string(CYPoolCString(pool, context, name));
Class _class(object_getClass(self));
-#ifdef __APPLE__
if (objc_property_t property = class_getProperty(_class, string)) {
PropertyAttributes attributes(property);
SEL sel(sel_registerName(attributes.Getter()));
return CYSendMessage(pool, context, self, NULL, sel, 0, NULL, false);
}
-#endif
if (CYHasImplicitProperties(_class))
if (SEL sel = sel_getUid(string))
@@ -1864,7 +1780,6 @@ static bool Instance_setProperty(JSContextRef context, JSObjectRef object, JSStr
const char *string(CYPoolCString(pool, context, name));
Class _class(object_getClass(self));
-#ifdef __APPLE__
if (objc_property_t property = class_getProperty(_class, string)) {
PropertyAttributes attributes(property);
if (const char *setter = attributes.Setter()) {
@@ -1874,7 +1789,6 @@ static bool Instance_setProperty(JSContextRef context, JSObjectRef object, JSStr
return true;
}
}
-#endif
size_t length(strlen(string));
@@ -1936,7 +1850,6 @@ static void Instance_getPropertyNames(JSContextRef context, JSObjectRef object,
CYPool pool;
Class _class(object_getClass(self));
-#ifdef __APPLE__
{
unsigned int size;
objc_property_t *data(class_copyPropertyList(_class, &size));
@@ -1944,21 +1857,14 @@ static void Instance_getPropertyNames(JSContextRef context, JSObjectRef object,
JSPropertyNameAccumulatorAddName(names, CYJSString(property_getName(data[i])));
free(data);
}
-#endif
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 {
@@ -1970,7 +1876,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) }
@@ -2091,6 +1997,7 @@ static JSValueRef Internal_getProperty(JSContextRef context, JSObjectRef object,
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);
@@ -2099,8 +2006,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);
}
}
@@ -2128,8 +2035,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;
}
}
@@ -2141,17 +2048,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) {
@@ -2178,11 +2079,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)));
@@ -2204,24 +2104,17 @@ 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
+#ifdef __APPLE__
static JSValueRef ObjectiveC_Image_Classes_getProperty(JSContextRef context, JSObjectRef object, JSStringRef property, JSValueRef *exception) { CYTry {
const char *internal(reinterpret_cast(JSObjectGetPrivate(object)));
@@ -2233,7 +2126,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;
@@ -2286,27 +2179,23 @@ 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) }
static void ObjectiveC_Protocols_getPropertyNames(JSContextRef context, JSObjectRef object, JSPropertyNameAccumulatorRef names) {
-#if OBJC_API_VERSION >= 2
unsigned int size;
Protocol **data(objc_copyProtocolList(&size));
for (size_t i(0); i != size; ++i)
JSPropertyNameAccumulatorAddName(names, CYJSString(protocol_getName(data[i])));
free(data);
-#else
- // XXX: fix this!
-#endif
}
static JSValueRef ObjectiveC_Constants_getProperty(JSContextRef context, JSObjectRef object, JSStringRef property, JSValueRef *exception) { CYTry {
CYPool pool;
CYUTF8String name(CYPoolUTF8String(pool, context, property));
if (name == "nil")
- return Instance::Make(context, nil);
+ return CYJSNull(context);
return NULL;
} CYCatch(NULL) }
@@ -2373,7 +2262,7 @@ static JSValueRef choose(JSContextRef context, JSObjectRef object, JSObjectRef _
CYGarbageCollect(context);
CYPool pool;
- Class _class(CYCastNSObject(&pool, context, arguments[0]));
+ id _class(CYCastNSObject(&pool, context, arguments[0]));
vm_address_t *zones(NULL);
unsigned size(0);
@@ -2435,6 +2324,8 @@ static bool stret(ffi_type *ffi_type) {
);
}
#endif
+#else
+#define CY_NO_STRET
#endif
JSValueRef CYSendMessage(CYPool &pool, JSContextRef context, id self, Class _class, SEL _cmd, size_t count, const JSValueRef arguments[], bool initialize) {
@@ -2452,11 +2343,14 @@ JSValueRef CYSendMessage(CYPool &pool, JSContextRef context, id self, Class _cla
imp = NULL;
CYPoolTry {
- NSMethodSignature *method([self methodSignatureForSelector:_cmd]);
- if (method == nil)
- throw CYJSError(context, "unrecognized selector %s sent to object %p", sel_getName(_cmd), self);
- type = CYPoolCString(pool, context, [method _typeString]);
+ if (NSMethodSignature *method = [self methodSignatureForSelector:_cmd])
+ type = CYPoolCString(pool, context, [method _typeString]);
+ else
+ type = NULL;
} CYPoolCatch(NULL)
+
+ if (type == NULL)
+ throw CYJSError(context, "unrecognized selector %s sent to object %p", sel_getName(_cmd), self);
}
void *setup[2];
@@ -2490,17 +2384,12 @@ JSValueRef CYSendMessage(CYPool &pool, JSContextRef context, id self, Class _cla
sig::sig_ffi_cif(pool, &sig::ObjectiveC, &signature, &cif);
if (imp == NULL) {
-#ifdef __APPLE__
#ifndef CY_NO_STRET
if (stret(cif.rtype))
imp = class_getMethodImplementation_stret(_class, _cmd);
else
#endif
imp = class_getMethodImplementation(_class, _cmd);
-#else
- objc_super super = {self, _class};
- imp = objc_msg_lookup_super(&super, _cmd);
-#endif
}
void (*function)() = reinterpret_cast(imp);
@@ -2592,7 +2481,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 {
@@ -2632,7 +2521,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 {
@@ -2690,6 +2579,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;
@@ -2732,16 +2622,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 {
@@ -2839,14 +2724,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");
@@ -2868,6 +2749,7 @@ void CYObjectiveC_Initialize() { /*XXX*/ JSContextRef context(NULL); CYPoolTry {
NSZombie_ = objc_getClass("_NSZombie_");
#else
NSBoolNumber_ = objc_getClass("NSBoolNumber");
+ NSZombie_ = objc_getClass("NSZombie");
#endif
JSClassDefinition definition;
@@ -2934,9 +2816,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);
@@ -2968,7 +2847,7 @@ void CYObjectiveC_Initialize() { /*XXX*/ JSContextRef context(NULL); CYPoolTry {
definition.getPropertyNames = &ObjectiveC_Constants_getPropertyNames;
ObjectiveC_Constants_ = JSClassCreate(&definition);
-#if OBJC_API_VERSION >= 2
+#ifdef __APPLE__
definition = kJSClassDefinitionEmpty;
definition.className = "ObjectiveC::Images";
definition.getProperty = &ObjectiveC_Images_getProperty;
@@ -3022,7 +2901,7 @@ void CYObjectiveC_SetupContext(JSContextRef context) { CYPoolTry {
CYSetProperty(context, ObjectiveC, CYJSString("constants"), constants);
CYArrayPush(context, alls, constants);
-#if OBJC_API_VERSION >= 2
+#ifdef __APPLE__
CYSetProperty(context, ObjectiveC, CYJSString("images"), JSObjectMake(context, ObjectiveC_Images_, NULL));
#endif
@@ -3092,7 +2971,16 @@ void CYObjectiveC_SetupContext(JSContextRef context) { CYPoolTry {
CYSetPrototype(context, CYCastJSObject(context, CYGetProperty(context, Selector, prototype_s)), Function_prototype);
} CYPoolCatch() }
-static CYHooks CYObjectiveCHooks = {
+static void *CYObjectiveC_CastSymbol(const char *name) {
+ if (false);
+#ifdef __GNU_LIBOBJC__
+ else if (strcmp(name, "object_getClass") == 0)
+ return reinterpret_cast(&object_getClass);
+#endif
+ return NULL;
+}
+
+static CYHook CYObjectiveCHook = {
&CYObjectiveC_ExecuteStart,
&CYObjectiveC_ExecuteEnd,
&CYObjectiveC_CallFunction,
@@ -3100,15 +2988,10 @@ static CYHooks CYObjectiveCHooks = {
&CYObjectiveC_SetupContext,
&CYObjectiveC_PoolFFI,
&CYObjectiveC_FromFFI,
+ &CYObjectiveC_CastSymbol,
};
-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);