490dd0c1984a5b8e6f6f24a79708cfb88251d0ef
[WebKit-https.git] / Tools / WebKitTestRunner / mac / PoseAsClass.mm
1 /*
2  * Copyright (C) 2013 Apple Inc. All rights reserved.
3  *
4  * Redistribution and use in source and binary forms, with or without
5  * modification, are permitted provided that the following conditions
6  * are met:
7  * 1. Redistributions of source code must retain the above copyright
8  *    notice, this list of conditions and the following disclaimer.
9  * 2. Redistributions in binary form must reproduce the above copyright
10  *    notice, this list of conditions and the following disclaimer in the
11  *    documentation and/or other materials provided with the distribution.
12  *
13  * THIS SOFTWARE IS PROVIDED BY APPLE INC. AND ITS CONTRIBUTORS ``AS IS''
14  * AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO,
15  * THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR
16  * PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL APPLE INC. OR ITS CONTRIBUTORS
17  * BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR
18  * CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF
19  * SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS
20  * INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN
21  * CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE)
22  * ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF
23  * THE POSSIBILITY OF SUCH DAMAGE.
24  */
25
26 #include "config.h"
27 #include "PoseAsClass.h"
28
29 #import <objc/runtime.h>
30 #import <wtf/Assertions.h>
31
32 static void swizzleAllMethods(Class imposter, Class original)
33 {
34     unsigned imposterMethodCount;
35     Method* imposterMethods = class_copyMethodList(imposter, &imposterMethodCount);
36
37     unsigned originalMethodCount;
38     Method* originalMethods = class_copyMethodList(original, &originalMethodCount);
39
40     for (unsigned i = 0; i < imposterMethodCount; i++) {
41         SEL imposterMethodName = method_getName(imposterMethods[i]);
42
43         // Attempt to add the method to the original class.  If it fails, the method already exists and we should
44         // instead exchange the implementations.
45         if (class_addMethod(original, imposterMethodName, method_getImplementation(imposterMethods[i]), method_getTypeEncoding(imposterMethods[i])))
46             continue;
47
48         unsigned j = 0;
49         for (; j < originalMethodCount; j++) {
50             SEL originalMethodName = method_getName(originalMethods[j]);
51             if (sel_isEqual(imposterMethodName, originalMethodName))
52                 break;
53         }
54
55         // If class_addMethod failed above then the method must exist on the original class.
56         ASSERT(j < originalMethodCount);
57         method_exchangeImplementations(imposterMethods[i], originalMethods[j]);
58     }
59
60     free(imposterMethods);
61     free(originalMethods);
62 }
63
64 void poseAsClass(const char* imposter, const char* original)
65 {
66     Class imposterClass = objc_getClass(imposter);
67     Class originalClass = objc_getClass(original);
68
69     // Swizzle instance methods
70     swizzleAllMethods(imposterClass, originalClass);
71     // and then class methods
72     swizzleAllMethods(object_getClass(imposterClass), object_getClass(originalClass));
73 }