1 /*
2 * Copyright (C) 2011 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
28 #include "wtf/RefCounted.h"
29 #include "wtf/Functional.h"
30 #include <gtest/gtest.h>
31
32 namespace {
33
returnFortyTwo()34 static int returnFortyTwo()
35 {
36 return 42;
37 }
38
TEST(FunctionalTest,Basic)39 TEST(FunctionalTest, Basic)
40 {
41 Function<int()> emptyFunction;
42 EXPECT_TRUE(emptyFunction.isNull());
43
44 Function<int()> returnFortyTwoFunction = bind(returnFortyTwo);
45 EXPECT_FALSE(returnFortyTwoFunction.isNull());
46 EXPECT_EQ(42, returnFortyTwoFunction());
47 }
48
multiplyByTwo(int n)49 static int multiplyByTwo(int n)
50 {
51 return n * 2;
52 }
53
multiplyByOneAndAHalf(double d)54 static double multiplyByOneAndAHalf(double d)
55 {
56 return d * 1.5;
57 }
58
TEST(FunctionalTest,UnaryBind)59 TEST(FunctionalTest, UnaryBind)
60 {
61 Function<int()> multiplyFourByTwoFunction = bind(multiplyByTwo, 4);
62 EXPECT_EQ(8, multiplyFourByTwoFunction());
63
64 Function<double()> multiplyByOneAndAHalfFunction = bind(multiplyByOneAndAHalf, 3);
65 EXPECT_EQ(4.5, multiplyByOneAndAHalfFunction());
66 }
67
TEST(FunctionalTest,UnaryPartBind)68 TEST(FunctionalTest, UnaryPartBind)
69 {
70 Function<int(int)> multiplyByTwoFunction = bind<int>(multiplyByTwo);
71 EXPECT_EQ(8, multiplyByTwoFunction(4));
72
73 Function<double(double)> multiplyByOneAndAHalfFunction = bind<double>(multiplyByOneAndAHalf);
74 EXPECT_EQ(4.5, multiplyByOneAndAHalfFunction(3));
75 }
76
multiply(int x,int y)77 static int multiply(int x, int y)
78 {
79 return x * y;
80 }
81
subtract(int x,int y)82 static int subtract(int x, int y)
83 {
84 return x - y;
85 }
86
TEST(FunctionalTest,BinaryBind)87 TEST(FunctionalTest, BinaryBind)
88 {
89 Function<int()> multiplyFourByTwoFunction = bind(multiply, 4, 2);
90 EXPECT_EQ(8, multiplyFourByTwoFunction());
91
92 Function<int()> subtractTwoFromFourFunction = bind(subtract, 4, 2);
93 EXPECT_EQ(2, subtractTwoFromFourFunction());
94 }
95
TEST(FunctionalTest,BinaryPartBind)96 TEST(FunctionalTest, BinaryPartBind)
97 {
98 Function<int(int)> multiplyFourFunction = bind<int>(multiply, 4);
99 EXPECT_EQ(8, multiplyFourFunction(2));
100 Function<int(int, int)> multiplyFunction = bind<int, int>(multiply);
101 EXPECT_EQ(8, multiplyFunction(4, 2));
102
103 Function<int(int)> subtractFromFourFunction = bind<int>(subtract, 4);
104 EXPECT_EQ(2, subtractFromFourFunction(2));
105 Function<int(int, int)> subtractFunction = bind<int, int>(subtract);
106 EXPECT_EQ(2, subtractFunction(4, 2));
107 }
108
sixArgFunc(int a,double b,char c,int * d,double * e,char * f)109 static void sixArgFunc(int a, double b, char c, int* d, double* e, char* f)
110 {
111 *d = a;
112 *e = b;
113 *f = c;
114 }
115
assertArgs(int actualInt,double actualDouble,char actualChar,int expectedInt,double expectedDouble,char expectedChar)116 static void assertArgs(int actualInt, double actualDouble, char actualChar, int expectedInt, double expectedDouble, char expectedChar)
117 {
118 EXPECT_EQ(expectedInt, actualInt);
119 EXPECT_EQ(expectedDouble, actualDouble);
120 EXPECT_EQ(expectedChar, actualChar);
121 }
122
TEST(FunctionalTest,MultiPartBind)123 TEST(FunctionalTest, MultiPartBind)
124 {
125 int a = 0;
126 double b = 0.5;
127 char c = 'a';
128
129 Function<void(int, double, char, int*, double*, char*)> unbound =
130 bind<int, double, char, int*, double*, char*>(sixArgFunc);
131 unbound(1, 1.5, 'b', &a, &b, &c);
132 assertArgs(a, b, c, 1, 1.5, 'b');
133
134 Function<void(double, char, int*, double*, char*)> oneBound =
135 bind<double, char, int*, double*, char*>(sixArgFunc, 2);
136 oneBound(2.5, 'c', &a, &b, &c);
137 assertArgs(a, b, c, 2, 2.5, 'c');
138
139 Function<void(char, int*, double*, char*)> twoBound =
140 bind<char, int*, double*, char*>(sixArgFunc, 3, 3.5);
141 twoBound('d', &a, &b, &c);
142 assertArgs(a, b, c, 3, 3.5, 'd');
143
144 Function<void(int*, double*, char*)> threeBound =
145 bind<int*, double*, char*>(sixArgFunc, 4, 4.5, 'e');
146 threeBound(&a, &b, &c);
147 assertArgs(a, b, c, 4, 4.5, 'e');
148
149 Function<void(double*, char*)> fourBound =
150 bind<double*, char*>(sixArgFunc, 5, 5.5, 'f', &a);
151 fourBound(&b, &c);
152 assertArgs(a, b, c, 5, 5.5, 'f');
153
154 Function<void(char*)> fiveBound =
155 bind<char*>(sixArgFunc, 6, 6.5, 'g', &a, &b);
156 fiveBound(&c);
157 assertArgs(a, b, c, 6, 6.5, 'g');
158
159 Function<void()> sixBound =
160 bind(sixArgFunc, 7, 7.5, 'h', &a, &b, &c);
161 sixBound();
162 assertArgs(a, b, c, 7, 7.5, 'h');
163 }
164
165 class A {
166 public:
A(int i)167 explicit A(int i)
168 : m_i(i)
169 {
170 }
171
f()172 int f() { return m_i; }
addF(int j)173 int addF(int j) { return m_i + j; }
174
175 private:
176 int m_i;
177 };
178
TEST(FunctionalTest,MemberFunctionBind)179 TEST(FunctionalTest, MemberFunctionBind)
180 {
181 A a(10);
182 Function<int()> function1 = bind(&A::f, &a);
183 EXPECT_EQ(10, function1());
184
185 Function<int()> function2 = bind(&A::addF, &a, 15);
186 EXPECT_EQ(25, function2());
187 }
188
TEST(FunctionalTest,MemberFunctionPartBind)189 TEST(FunctionalTest, MemberFunctionPartBind)
190 {
191 A a(10);
192 Function<int(class A*)> function1 = bind<class A*>(&A::f);
193 EXPECT_EQ(10, function1(&a));
194
195 Function<int(class A*, int)> unboundFunction2 =
196 bind<class A*, int>(&A::addF);
197 EXPECT_EQ(25, unboundFunction2(&a, 15));
198 Function<int(int)> objectBoundFunction2 =
199 bind<int>(&A::addF, &a);
200 EXPECT_EQ(25, objectBoundFunction2(15));
201 }
202
203 class Number : public RefCounted<Number> {
204 public:
create(int value)205 static PassRefPtr<Number> create(int value)
206 {
207 return adoptRef(new Number(value));
208 }
209
~Number()210 ~Number()
211 {
212 m_value = 0;
213 }
214
value() const215 int value() const { return m_value; }
216
217 private:
Number(int value)218 explicit Number(int value)
219 : m_value(value)
220 {
221 }
222
223 int m_value;
224 };
225
multiplyNumberByTwo(Number * number)226 static int multiplyNumberByTwo(Number* number)
227 {
228 return number->value() * 2;
229 }
230
TEST(FunctionalTest,RefCountedStorage)231 TEST(FunctionalTest, RefCountedStorage)
232 {
233 RefPtr<Number> five = Number::create(5);
234 Function<int()> multiplyFiveByTwoFunction = bind(multiplyNumberByTwo, five);
235 EXPECT_EQ(10, multiplyFiveByTwoFunction());
236
237 Function<int()> multiplyFourByTwoFunction = bind(multiplyNumberByTwo, Number::create(4));
238 EXPECT_EQ(8, multiplyFourByTwoFunction());
239
240 RefPtr<Number> six = Number::create(6);
241 Function<int()> multiplySixByTwoFunction = bind(multiplyNumberByTwo, six.release());
242 EXPECT_FALSE(six);
243 EXPECT_EQ(12, multiplySixByTwoFunction());
244 }
245
246 } // namespace
247