-
Notifications
You must be signed in to change notification settings - Fork 2
/
Copy pathMethodSwizzle.m
123 lines (110 loc) · 4.3 KB
/
MethodSwizzle.m
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
//
// MethodSwizzle.m
//
// Copyright (c) 2006 Tildesoft. All rights reserved.
//
// Permission is hereby granted, free of charge, to any person obtaining a
// copy of this software and associated documentation files (the "Software"),
// to deal in the Software without restriction, including without limitation
// the rights to use, copy, modify, merge, publish, distribute, sublicense,
// and/or sell copies of the Software, and to permit persons to whom the
// Software is furnished to do so, subject to the following conditions:
//
// The above copyright notice and this permission notice shall be included in
// all copies or substantial portions of the Software.
//
// THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
// IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
// FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
// AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
// LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING
// FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER
// DEALINGS IN THE SOFTWARE.
// Implementation of Method Swizzling, inspired by
// http://www.cocoadev.com/index.pl?MethodSwizzling
// solves the inherited method problem
#import "MethodSwizzle.h"
#import <stdlib.h>
#import <string.h>
#import <objc/objc-runtime.h>
#import <objc/objc-class.h>
#import <uuid/uuid.h>
static BOOL _PerformSwizzle(Class klass, SEL origSel, SEL altSel, BOOL forInstance);
BOOL ClassMethodSwizzle(Class klass, SEL origSel, SEL altSel) {
return _PerformSwizzle(klass, origSel, altSel, NO);
}
BOOL MethodSwizzle(Class klass, SEL origSel, SEL altSel) {
return _PerformSwizzle(klass, origSel, altSel, YES);
}
// if the origSel isn't present in the class, pull it up from where it exists
// then do the swizzle
BOOL _PerformSwizzle(Class klass, SEL origSel, SEL altSel, BOOL forInstance) {
// First, make sure the class isn't nil
if (klass != nil) {
Method origMethod = NULL, altMethod = NULL;
// Next, look for the methods
Class iterKlass = (forInstance ? klass : klass->isa);
void *iterator = NULL;
struct objc_method_list *mlist = class_nextMethodList(iterKlass, &iterator);
while (mlist != NULL) {
int i;
for (i = 0; i < mlist->method_count; ++i) {
if (mlist->method_list[i].method_name == origSel) {
origMethod = &mlist->method_list[i];
break;
}
if (mlist->method_list[i].method_name == altSel) {
altMethod = &mlist->method_list[i];
break;
}
}
mlist = class_nextMethodList(iterKlass, &iterator);
}
if (origMethod == NULL || altMethod == NULL) {
// one or both methods are not in the immediate class
// try searching the entire hierarchy
// remember, iterKlass is the class we care about - klass || klass->isa
// class_getInstanceMethod on a metaclass is the same as class_getClassMethod on the real class
BOOL pullOrig = NO, pullAlt = NO;
if (origMethod == NULL) {
origMethod = class_getInstanceMethod(iterKlass, origSel);
pullOrig = YES;
}
if (altMethod == NULL) {
altMethod = class_getInstanceMethod(iterKlass, altSel);
pullAlt = YES;
}
// die now if one of the methods doesn't exist anywhere in the hierarchy
// this way we won't make any changes to the class if we can't finish
if (origMethod == NULL || altMethod == NULL) {
NSLog(@"origMethod: %d altMethod: %d", origMethod, altMethod);
return NO;
}
// we can safely assume one of the two methods, at least, will be pulled
// pull them up
size_t listSize = sizeof(struct objc_method_list);
if (pullOrig && pullAlt) listSize += sizeof(struct objc_method); // need 2 methods
struct objc_method_list *mlist = malloc(listSize);
mlist->obsolete = NULL;
int i = 0;
if (pullOrig) {
memcpy(&mlist->method_list[i], origMethod, sizeof(struct objc_method));
origMethod = &mlist->method_list[i];
i++;
}
if (pullAlt) {
memcpy(&mlist->method_list[i], altMethod, sizeof(struct objc_method));
altMethod = &mlist->method_list[i];
i++;
}
mlist->method_count = i;
class_addMethods(iterKlass, mlist);
}
// now swizzle
IMP temp = origMethod->method_imp;
origMethod->method_imp = altMethod->method_imp;
altMethod->method_imp = temp;
return YES;
}
return NO;
}