substrate2.h 13 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288
  1. #ifndef SUBSTRATE_H_
  2. #define SUBSTRATE_H_
  3. #ifdef __APPLE__
  4. #ifdef __cplusplus
  5. extern "C" {
  6. #endif
  7. #include <mach-o/nlist.h>
  8. #ifdef __cplusplus
  9. }
  10. #endif
  11. #include <objc/runtime.h>
  12. #include <objc/message.h>
  13. #endif
  14. #include <dlfcn.h>
  15. #define _finline \
  16. inline __attribute__((always_inline))
  17. #define _disused \
  18. __attribute__((unused))
  19. #ifdef __cplusplus
  20. #define _default(value) = value
  21. #else
  22. #define _default(value)
  23. #endif
  24. #ifdef __cplusplus
  25. extern "C" {
  26. #endif
  27. void MSHookFunction(void *symbol, void *replace, void **result);
  28. #ifdef __APPLE__
  29. IMP MSHookMessage(Class _class, SEL sel, IMP imp, const char *prefix _default(NULL));
  30. void MSHookMessageEx(Class _class, SEL sel, IMP imp, IMP *result);
  31. #endif
  32. #ifdef __cplusplus
  33. }
  34. #endif
  35. #ifdef __cplusplus
  36. #ifdef __APPLE__
  37. namespace etl {
  38. template <unsigned Case_>
  39. struct Case {
  40. static char value[Case_ + 1];
  41. };
  42. typedef Case<true> Yes;
  43. typedef Case<false> No;
  44. namespace be {
  45. template <typename Checked_>
  46. static Yes CheckClass_(void (Checked_::*)());
  47. template <typename Checked_>
  48. static No CheckClass_(...);
  49. }
  50. template <typename Type_>
  51. struct IsClass {
  52. void gcc32();
  53. static const bool value = (sizeof(be::CheckClass_<Type_>(0).value) == sizeof(Yes::value));
  54. };
  55. }
  56. template <typename Type_>
  57. static inline Type_ *MSHookMessage(Class _class, SEL sel, Type_ *imp, const char *prefix = NULL) {
  58. return reinterpret_cast<Type_ *>(MSHookMessage(_class, sel, reinterpret_cast<IMP>(imp), prefix));
  59. }
  60. template <typename Type_>
  61. static inline void MSHookMessage(Class _class, SEL sel, Type_ *imp, Type_ **result) {
  62. return MSHookMessageEx(_class, sel, reinterpret_cast<IMP>(imp), reinterpret_cast<IMP *>(result));
  63. }
  64. template <typename Type_>
  65. static inline Type_ &MSHookIvar(id self, const char *name) {
  66. Ivar ivar(class_getInstanceVariable(object_getClass(self), name));
  67. void *pointer(ivar == NULL ? NULL : reinterpret_cast<char *>(self) + ivar_getOffset(ivar));
  68. return *reinterpret_cast<Type_ *>(pointer);
  69. }
  70. #define MSAddMessage0(_class, type, arg0) \
  71. class_addMethod($ ## _class, @selector(arg0), (IMP) &$ ## _class ## $ ## arg0, type);
  72. #define MSAddMessage1(_class, type, arg0) \
  73. class_addMethod($ ## _class, @selector(arg0:), (IMP) &$ ## _class ## $ ## arg0 ## $, type);
  74. #define MSAddMessage2(_class, type, arg0, arg1) \
  75. class_addMethod($ ## _class, @selector(arg0:arg1:), (IMP) &$ ## _class ## $ ## arg0 ## $ ## arg1 ## $, type);
  76. #define MSHookMessage0(_class, arg0) \
  77. MSHookMessage($ ## _class, @selector(arg0), MSHake(_class ## $ ## arg0))
  78. #define MSHookMessage1(_class, arg0) \
  79. MSHookMessage($ ## _class, @selector(arg0:), MSHake(_class ## $ ## arg0 ## $))
  80. #define MSHookMessage2(_class, arg0, arg1) \
  81. MSHookMessage($ ## _class, @selector(arg0:arg1:), MSHake(_class ## $ ## arg0 ## $ ## arg1 ## $))
  82. #define MSHookMessage3(_class, arg0, arg1, arg2) \
  83. MSHookMessage($ ## _class, @selector(arg0:arg1:arg2:), MSHake(_class ## $ ## arg0 ## $ ## arg1 ## $ ## arg2 ## $))
  84. #define MSHookMessage4(_class, arg0, arg1, arg2, arg3) \
  85. MSHookMessage($ ## _class, @selector(arg0:arg1:arg2:arg3:), MSHake(_class ## $ ## arg0 ## $ ## arg1 ## $ ## arg2 ## $ ## arg3 ## $))
  86. #define MSHookMessage5(_class, arg0, arg1, arg2, arg3, arg4) \
  87. MSHookMessage($ ## _class, @selector(arg0:arg1:arg2:arg3:arg4:), MSHake(_class ## $ ## arg0 ## $ ## arg1 ## $ ## arg2 ## $ ## arg3 ## $ ## arg4 ## $))
  88. #define MSHookMessage6(_class, arg0, arg1, arg2, arg3, arg4, arg5) \
  89. MSHookMessage($ ## _class, @selector(arg0:arg1:arg2:arg3:arg4:arg5:), MSHake(_class ## $ ## arg0 ## $ ## arg1 ## $ ## arg2 ## $ ## arg3 ## $ ## arg4 ## $ ## arg5 ## $))
  90. #define MSRegister_(name, dollar, colon) \
  91. static class C_$ ## name ## $ ## dollar { public: _finline C_$ ## name ## $ ##dollar() { \
  92. MSHookMessage($ ## name, @selector(colon), MSHake(name ## $ ## dollar)); \
  93. } } V_$ ## dollar; \
  94. #define MSIgnore_(name, dollar, colon)
  95. #define MSMessage_(extra, type, _class, name, dollar, colon, call, args...) \
  96. static type _$ ## name ## $ ## dollar(Class _cls, type (*_old)(_class, SEL, ## args), type (*_spr)(struct objc_super *, SEL, ## args), _class self, SEL _cmd, ## args); \
  97. MSHook(type, name ## $ ## dollar, _class self, SEL _cmd, ## args) { \
  98. Class const _cls($ ## name); \
  99. type (* const _old)(_class, SEL, ## args) = _ ## name ## $ ## dollar; \
  100. typedef type (*msgSendSuper_t)(struct objc_super *, SEL, ## args); \
  101. msgSendSuper_t const _spr(::etl::IsClass<type>::value ? reinterpret_cast<msgSendSuper_t>(&objc_msgSendSuper_stret) : reinterpret_cast<msgSendSuper_t>(&objc_msgSendSuper)); \
  102. return _$ ## name ## $ ## dollar call; \
  103. } \
  104. extra(name, dollar, colon) \
  105. static _finline type _$ ## name ## $ ## dollar(Class _cls, type (*_old)(_class, SEL, ## args), type (*_spr)(struct objc_super *, SEL, ## args), _class self, SEL _cmd, ## args)
  106. /* for((x=1;x!=7;++x)){ echo -n "#define MSMessage${x}_(extra, type, _class, name";for((y=0;y!=x;++y));do echo -n ", sel$y";done;for((y=0;y!=x;++y));do echo -n ", type$y, arg$y";done;echo ") \\";echo -n " MSMessage_(extra, type, _class, name,";for((y=0;y!=x;++y));do if [[ $y -ne 0 ]];then echo -n " ##";fi;echo -n " sel$y ## $";done;echo -n ", ";for((y=0;y!=x;++y));do echo -n "sel$y:";done;echo -n ", (_cls, _old, _spr, self, _cmd";for((y=0;y!=x;++y));do echo -n ", arg$y";done;echo -n ")";for((y=0;y!=x;++y));do echo -n ", type$y arg$y";done;echo ")";} */
  107. #define MSMessage0_(extra, type, _class, name, sel0) \
  108. MSMessage_(extra, type, _class, name, sel0, sel0, (_cls, _old, _spr, self, _cmd))
  109. #define MSMessage1_(extra, type, _class, name, sel0, type0, arg0) \
  110. MSMessage_(extra, type, _class, name, sel0 ## $, sel0:, (_cls, _old, _spr, self, _cmd, arg0), type0 arg0)
  111. #define MSMessage2_(extra, type, _class, name, sel0, sel1, type0, arg0, type1, arg1) \
  112. MSMessage_(extra, type, _class, name, sel0 ## $ ## sel1 ## $, sel0:sel1:, (_cls, _old, _spr, self, _cmd, arg0, arg1), type0 arg0, type1 arg1)
  113. #define MSMessage3_(extra, type, _class, name, sel0, sel1, sel2, type0, arg0, type1, arg1, type2, arg2) \
  114. MSMessage_(extra, type, _class, name, sel0 ## $ ## sel1 ## $ ## sel2 ## $, sel0:sel1:sel2:, (_cls, _old, _spr, self, _cmd, arg0, arg1, arg2), type0 arg0, type1 arg1, type2 arg2)
  115. #define MSMessage4_(extra, type, _class, name, sel0, sel1, sel2, sel3, type0, arg0, type1, arg1, type2, arg2, type3, arg3) \
  116. MSMessage_(extra, type, _class, name, sel0 ## $ ## sel1 ## $ ## sel2 ## $ ## sel3 ## $, sel0:sel1:sel2:sel3:, (_cls, _old, _spr, self, _cmd, arg0, arg1, arg2, arg3), type0 arg0, type1 arg1, type2 arg2, type3 arg3)
  117. #define MSMessage5_(extra, type, _class, name, sel0, sel1, sel2, sel3, sel4, type0, arg0, type1, arg1, type2, arg2, type3, arg3, type4, arg4) \
  118. MSMessage_(extra, type, _class, name, sel0 ## $ ## sel1 ## $ ## sel2 ## $ ## sel3 ## $ ## sel4 ## $, sel0:sel1:sel2:sel3:sel4:, (_cls, _old, _spr, self, _cmd, arg0, arg1, arg2, arg3, arg4), type0 arg0, type1 arg1, type2 arg2, type3 arg3, type4 arg4)
  119. #define MSMessage6_(extra, type, _class, name, sel0, sel1, sel2, sel3, sel4, sel5, type0, arg0, type1, arg1, type2, arg2, type3, arg3, type4, arg4, type5, arg5) \
  120. MSMessage_(extra, type, _class, name, sel0 ## $ ## sel1 ## $ ## sel2 ## $ ## sel3 ## $ ## sel4 ## $ ## sel5 ## $, sel0:sel1:sel2:sel3:sel4:sel5:, (_cls, _old, _spr, self, _cmd, arg0, arg1, arg2, arg3, arg4, arg5), type0 arg0, type1 arg1, type2 arg2, type3 arg3, type4 arg4, type5 arg5)
  121. #define MSInstanceMessage0(type, _class, args...) MSMessage0_(MSIgnore_, type, _class *, _class, ## args)
  122. #define MSInstanceMessage1(type, _class, args...) MSMessage1_(MSIgnore_, type, _class *, _class, ## args)
  123. #define MSInstanceMessage2(type, _class, args...) MSMessage2_(MSIgnore_, type, _class *, _class, ## args)
  124. #define MSInstanceMessage3(type, _class, args...) MSMessage3_(MSIgnore_, type, _class *, _class, ## args)
  125. #define MSInstanceMessage4(type, _class, args...) MSMessage4_(MSIgnore_, type, _class *, _class, ## args)
  126. #define MSInstanceMessage5(type, _class, args...) MSMessage5_(MSIgnore_, type, _class *, _class, ## args)
  127. #define MSInstanceMessage6(type, _class, args...) MSMessage6_(MSIgnore_, type, _class *, _class, ## args)
  128. #define MSClassMessage0(type, _class, args...) MSMessage0_(MSIgnore_, type, Class, $ ## _class, ## args)
  129. #define MSClassMessage1(type, _class, args...) MSMessage1_(MSIgnore_, type, Class, $ ## _class, ## args)
  130. #define MSClassMessage2(type, _class, args...) MSMessage2_(MSIgnore_, type, Class, $ ## _class, ## args)
  131. #define MSClassMessage3(type, _class, args...) MSMessage3_(MSIgnore_, type, Class, $ ## _class, ## args)
  132. #define MSClassMessage4(type, _class, args...) MSMessage4_(MSIgnore_, type, Class, $ ## _class, ## args)
  133. #define MSClassMessage5(type, _class, args...) MSMessage5_(MSIgnore_, type, Class, $ ## _class, ## args)
  134. #define MSClassMessage6(type, _class, args...) MSMessage6_(MSIgnore_, type, Class, $ ## _class, ## args)
  135. #define MSInstanceMessageHook0(type, _class, args...) MSMessage0_(MSRegister_, type, _class *, _class, ## args)
  136. #define MSInstanceMessageHook1(type, _class, args...) MSMessage1_(MSRegister_, type, _class *, _class, ## args)
  137. #define MSInstanceMessageHook2(type, _class, args...) MSMessage2_(MSRegister_, type, _class *, _class, ## args)
  138. #define MSInstanceMessageHook3(type, _class, args...) MSMessage3_(MSRegister_, type, _class *, _class, ## args)
  139. #define MSInstanceMessageHook4(type, _class, args...) MSMessage4_(MSRegister_, type, _class *, _class, ## args)
  140. #define MSInstanceMessageHook5(type, _class, args...) MSMessage5_(MSRegister_, type, _class *, _class, ## args)
  141. #define MSInstanceMessageHook6(type, _class, args...) MSMessage6_(MSRegister_, type, _class *, _class, ## args)
  142. #define MSClassMessageHook0(type, _class, args...) MSMessage0_(MSRegister_, type, Class, $ ## _class, ## args)
  143. #define MSClassMessageHook1(type, _class, args...) MSMessage1_(MSRegister_, type, Class, $ ## _class, ## args)
  144. #define MSClassMessageHook2(type, _class, args...) MSMessage2_(MSRegister_, type, Class, $ ## _class, ## args)
  145. #define MSClassMessageHook3(type, _class, args...) MSMessage3_(MSRegister_, type, Class, $ ## _class, ## args)
  146. #define MSClassMessageHook4(type, _class, args...) MSMessage4_(MSRegister_, type, Class, $ ## _class, ## args)
  147. #define MSClassMessageHook5(type, _class, args...) MSMessage5_(MSRegister_, type, Class, $ ## _class, ## args)
  148. #define MSClassMessageHook6(type, _class, args...) MSMessage6_(MSRegister_, type, Class, $ ## _class, ## args)
  149. #define MSOldCall(args...) \
  150. _old(self, _cmd, ## args)
  151. #define MSSuperCall(args...) \
  152. _spr(& (struct objc_super) {self, class_getSuperclass(_cls)}, _cmd, ## args)
  153. #define MSIvarHook(type, name) \
  154. type &name(MSHookIvar<type>(self, #name))
  155. #define MSClassHook(name) \
  156. static Class $ ## name = objc_getClass(#name);
  157. #define MSMetaClassHook(name) \
  158. static Class $$ ## name = object_getClass($ ## name);
  159. #endif/*__APPLE__*/
  160. template <typename Type_>
  161. static inline void MSHookFunction(Type_ *symbol, Type_ *replace, Type_ **result) {
  162. return MSHookFunction(
  163. reinterpret_cast<void *>(symbol),
  164. reinterpret_cast<void *>(replace),
  165. reinterpret_cast<void **>(result)
  166. );
  167. }
  168. template <typename Type_>
  169. static inline void MSHookFunction(Type_ *symbol, Type_ *replace) {
  170. return MSHookFunction(symbol, replace, reinterpret_cast<Type_ **>(NULL));
  171. }
  172. template <typename Type_>
  173. static inline void MSHookSymbol(Type_ *&value, const char *name, void *handle = RTLD_DEFAULT) {
  174. value = reinterpret_cast<Type_ *>(dlsym(handle, name));
  175. }
  176. /* Objective-C Handle<> {{{ */
  177. template <typename Type_>
  178. class _H {
  179. typedef _H<Type_> This_;
  180. private:
  181. Type_ *value_;
  182. _finline void Retain_() {
  183. if (value_ != nil)
  184. [value_ retain];
  185. }
  186. _finline void Clear_() {
  187. if (value_ != nil)
  188. [value_ release];
  189. }
  190. public:
  191. _finline _H(const This_ &rhs) :
  192. value_(rhs.value_ == nil ? nil : [rhs.value_ retain])
  193. {
  194. }
  195. _finline _H(Type_ *value = NULL, bool mended = false) :
  196. value_(value)
  197. {
  198. if (!mended)
  199. Retain_();
  200. }
  201. _finline ~_H() {
  202. Clear_();
  203. }
  204. _finline operator Type_ *() const {
  205. return value_;
  206. }
  207. _finline This_ &operator =(Type_ *value) {
  208. if (value_ != value) {
  209. Type_ *old(value_);
  210. value_ = value;
  211. Retain_();
  212. if (old != nil)
  213. [old release];
  214. } return *this;
  215. }
  216. };
  217. /* }}} */
  218. #define _pooled _H<NSAutoreleasePool> _pool([[NSAutoreleasePool alloc] init], true);
  219. #endif
  220. #define MSHook(type, name, args...) \
  221. _disused static type (*_ ## name)(args); \
  222. static type $ ## name(args)
  223. #define MSHake(name) \
  224. &$ ## name, &_ ## name
  225. #define MSInitialize \
  226. __attribute__((constructor)) static void _MSInitialize(void)
  227. #define Foundation_f "/System/Library/Frameworks/Foundation.framework/Foundation"
  228. #define UIKit_f "/System/Library/Frameworks/UIKit.framework/UIKit"
  229. #define JavaScriptCore_f "/System/Library/PrivateFrameworks/JavaScriptCore.framework/JavaScriptCore"
  230. #define IOKit_f "/System/Library/Frameworks/IOKit.framework/IOKit"
  231. #endif//SUBSTRATE_H_