X-Git-Url: https://git.saurik.com/cycript.git/blobdiff_plain/88c31c1e00d8a528f46aabe2e4886a78c8eedd3b..1e8d80477a3e058a30c477955f1e0c56deb6e956:/Internal.hpp?ds=sidebyside diff --git a/Internal.hpp b/Internal.hpp index b4acd18..7e6b1fd 100644 --- a/Internal.hpp +++ b/Internal.hpp @@ -34,20 +34,30 @@ #include "Pooling.hpp" #include "Utility.hpp" +struct CYPropertyName; + JSGlobalContextRef CYGetJSContext(JSContextRef context); sig::Type *Structor_(CYPool &pool, sig::Aggregate *aggregate); -extern JSClassRef Functor_; - -template -struct CYPrivate : +struct CYRoot : CYData { - static JSClassRef Class_; + // XXX: without this, CYData is zero-initialized?! + CYRoot() : + CYData() + { + } _finline JSValueRef GetPrototype(JSContextRef context) const { return NULL; } +}; + +template +struct CYPrivateOld : + Base_ +{ + static JSClassRef Class_; template _finline static JSClassRef GetClass(Args_ &&... args) { @@ -69,8 +79,51 @@ struct CYPrivate : } }; +template +JSClassRef CYPrivateOld::Class_; + +template +struct CYPrivate { + static JSClassRef Class_; + + template + static JSObjectRef Make(JSContextRef context, Args_ &&... args) { + Internal_ *internal(new Internal_(cy::Forward(args)...)); + JSObjectRef object(JSObjectMake(context, Class_, internal)); + if (JSValueRef prototype = internal->GetPrototype(context)) + CYSetPrototype(context, object, prototype); + return object; + } + + template + static JSObjectRef Cache(JSContextRef context, Arg_ *arg) { + JSObjectRef global(CYGetGlobalObject(context)); + JSObjectRef cy(CYCastJSObject(context, CYGetProperty(context, global, cy_s))); + + char label[32]; + sprintf(label, "%s%p", Internal_::Cache_, arg); + CYJSString name(label); + + JSValueRef value(CYGetProperty(context, cy, name)); + if (!JSValueIsUndefined(context, value)) + return CYCastJSObject(context, value); + + JSObjectRef object(Make(context, arg)); + CYSetProperty(context, cy, name, object); + return object; + } + + static Internal_ *Get(JSContextRef context, JSObjectRef object) { + _assert(JSValueIsObjectOfClass(context, object, Class_)); + return static_cast(JSObjectGetPrivate(object)); + } +}; + +template +JSClassRef CYPrivate::Class_; + struct Type_privateData : - CYPrivate + CYRoot { ffi_type *ffi_; sig::Type *type_; @@ -117,9 +170,6 @@ struct Type_privateData : } }; -template -JSClassRef CYPrivate::Class_; - struct CYProtect { private: JSGlobalContextRef context_; @@ -154,10 +204,40 @@ struct CYProtect { } }; +class CYBuffer { + private: + JSObjectRef owner_; + CYPool *pool_; + + public: + CYBuffer(JSContextRef context) : + owner_(CYPrivate::Make(context)), + pool_(CYPrivate::Get(context, owner_)->pool_) + { + auto internal(CYPrivate::Get(context, owner_)); + internal->pool_->malloc(10); + } + + operator JSObjectRef() const { + return owner_; + } + + operator CYPool *() const { + return pool_; + } + + CYPool *operator ->() const { + return pool_; + } +}; + namespace cy { struct Functor : - CYPrivate + CYRoot { + public: + static JSClassRef Class_; + private: void set() { sig::sig_ffi_cif(*pool_, variadic_ ? signature_.count : 0, signature_, &cif_); @@ -185,8 +265,7 @@ struct Functor : set(); } - static JSStaticFunction const * const StaticFunctions; - static JSStaticValue const * const StaticValues; + virtual CYPropertyName *GetName(CYPool &pool) const; }; } struct Closure_privateData :