blob: c5e8fec4f06d0787ddb452e8a21490d9c3f4ebc2 [file] [log] [blame]
/*
* Copyright (C) 2011 Apple Inc. All rights reserved.
*
* Redistribution and use in source and binary forms, with or without
* modification, are permitted provided that the following conditions
* are met:
* 1. Redistributions of source code must retain the above copyright
* notice, this list of conditions and the following disclaimer.
* 2. Redistributions in binary form must reproduce the above copyright
* notice, this list of conditions and the following disclaimer in the
* documentation and/or other materials provided with the distribution.
*
* THIS SOFTWARE IS PROVIDED BY APPLE INC. AND ITS CONTRIBUTORS ``AS IS''
* AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO,
* THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR
* PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL APPLE INC. OR ITS CONTRIBUTORS
* BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR
* CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF
* SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS
* INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN
* CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE)
* ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF
* THE POSSIBILITY OF SUCH DAMAGE.
*/
#include "config.h"
#include "wtf/Functional.h"
#include "wtf/OwnPtr.h"
#include "wtf/RefCounted.h"
#include <gtest/gtest.h>
namespace WTF {
class UnwrappedClass {
public:
explicit UnwrappedClass(int value)
: m_value(value)
{
}
int value() const { return m_value; }
private:
int m_value;
};
class WrappedClass {
public:
explicit WrappedClass(int value)
: m_value(value)
{
}
UnwrappedClass unwrap() const { return UnwrappedClass(m_value); }
private:
int m_value;
};
// This class must be wrapped in bind() and unwrapped in closure execution.
class ClassToBeWrapped {
WTF_MAKE_NONCOPYABLE(ClassToBeWrapped);
public:
explicit ClassToBeWrapped(int value)
: m_value(value)
{
}
WrappedClass wrap() const { return WrappedClass(m_value); }
private:
int m_value;
};
template<> struct ParamStorageTraits<ClassToBeWrapped> {
using StorageType = WrappedClass;
static StorageType wrap(const ClassToBeWrapped& value) { return value.wrap(); }
static UnwrappedClass unwrap(const StorageType& value) { return value.unwrap(); }
};
namespace {
int returnFortyTwo()
{
return 42;
}
TEST(FunctionalTest, Basic)
{
OwnPtr<Function<int()>> returnFortyTwoFunction = bind(returnFortyTwo);
EXPECT_EQ(42, (*returnFortyTwoFunction)());
}
int multiplyByTwo(int n)
{
return n * 2;
}
double multiplyByOneAndAHalf(double d)
{
return d * 1.5;
}
TEST(FunctionalTest, UnaryBind)
{
OwnPtr<Function<int()>> multiplyFourByTwoFunction = bind(multiplyByTwo, 4);
EXPECT_EQ(8, (*multiplyFourByTwoFunction)());
OwnPtr<Function<double()>> multiplyByOneAndAHalfFunction = bind(multiplyByOneAndAHalf, 3);
EXPECT_EQ(4.5, (*multiplyByOneAndAHalfFunction)());
}
TEST(FunctionalTest, UnaryPartBind)
{
OwnPtr<Function<int(int)>> multiplyByTwoFunction = bind<int>(multiplyByTwo);
EXPECT_EQ(8, (*multiplyByTwoFunction)(4));
OwnPtr<Function<double(double)>> multiplyByOneAndAHalfFunction = bind<double>(multiplyByOneAndAHalf);
EXPECT_EQ(4.5, (*multiplyByOneAndAHalfFunction)(3));
}
int multiply(int x, int y)
{
return x * y;
}
int subtract(int x, int y)
{
return x - y;
}
TEST(FunctionalTest, BinaryBind)
{
OwnPtr<Function<int()>> multiplyFourByTwoFunction = bind(multiply, 4, 2);
EXPECT_EQ(8, (*multiplyFourByTwoFunction)());
OwnPtr<Function<int()>> subtractTwoFromFourFunction = bind(subtract, 4, 2);
EXPECT_EQ(2, (*subtractTwoFromFourFunction)());
}
TEST(FunctionalTest, BinaryPartBind)
{
OwnPtr<Function<int(int)>> multiplyFourFunction = bind<int>(multiply, 4);
EXPECT_EQ(8, (*multiplyFourFunction)(2));
OwnPtr<Function<int(int, int)>> multiplyFunction = bind<int, int>(multiply);
EXPECT_EQ(8, (*multiplyFunction)(4, 2));
OwnPtr<Function<int(int)>> subtractFromFourFunction = bind<int>(subtract, 4);
EXPECT_EQ(2, (*subtractFromFourFunction)(2));
OwnPtr<Function<int(int, int)>> subtractFunction = bind<int, int>(subtract);
EXPECT_EQ(2, (*subtractFunction)(4, 2));
}
void sixArgFunc(int a, double b, char c, int* d, double* e, char* f)
{
*d = a;
*e = b;
*f = c;
}
void assertArgs(int actualInt, double actualDouble, char actualChar, int expectedInt, double expectedDouble, char expectedChar)
{
EXPECT_EQ(expectedInt, actualInt);
EXPECT_EQ(expectedDouble, actualDouble);
EXPECT_EQ(expectedChar, actualChar);
}
TEST(FunctionalTest, MultiPartBind)
{
int a = 0;
double b = 0.5;
char c = 'a';
OwnPtr<Function<void(int, double, char, int*, double*, char*)>> unbound =
bind<int, double, char, int*, double*, char*>(sixArgFunc);
(*unbound)(1, 1.5, 'b', &a, &b, &c);
assertArgs(a, b, c, 1, 1.5, 'b');
OwnPtr<Function<void(double, char, int*, double*, char*)>> oneBound =
bind<double, char, int*, double*, char*>(sixArgFunc, 2);
(*oneBound)(2.5, 'c', &a, &b, &c);
assertArgs(a, b, c, 2, 2.5, 'c');
OwnPtr<Function<void(char, int*, double*, char*)>> twoBound =
bind<char, int*, double*, char*>(sixArgFunc, 3, 3.5);
(*twoBound)('d', &a, &b, &c);
assertArgs(a, b, c, 3, 3.5, 'd');
OwnPtr<Function<void(int*, double*, char*)>> threeBound =
bind<int*, double*, char*>(sixArgFunc, 4, 4.5, 'e');
(*threeBound)(&a, &b, &c);
assertArgs(a, b, c, 4, 4.5, 'e');
OwnPtr<Function<void(double*, char*)>> fourBound =
bind<double*, char*>(sixArgFunc, 5, 5.5, 'f', &a);
(*fourBound)(&b, &c);
assertArgs(a, b, c, 5, 5.5, 'f');
OwnPtr<Function<void(char*)>> fiveBound =
bind<char*>(sixArgFunc, 6, 6.5, 'g', &a, &b);
(*fiveBound)(&c);
assertArgs(a, b, c, 6, 6.5, 'g');
OwnPtr<Function<void()>> sixBound =
bind(sixArgFunc, 7, 7.5, 'h', &a, &b, &c);
(*sixBound)();
assertArgs(a, b, c, 7, 7.5, 'h');
}
class A {
public:
explicit A(int i)
: m_i(i)
{
}
int f() { return m_i; }
int addF(int j) { return m_i + j; }
private:
int m_i;
};
TEST(FunctionalTest, MemberFunctionBind)
{
A a(10);
OwnPtr<Function<int()>> function1 = bind(&A::f, &a);
EXPECT_EQ(10, (*function1)());
OwnPtr<Function<int()>> function2 = bind(&A::addF, &a, 15);
EXPECT_EQ(25, (*function2)());
}
TEST(FunctionalTest, MemberFunctionPartBind)
{
A a(10);
OwnPtr<Function<int(class A*)>> function1 = bind<class A*>(&A::f);
EXPECT_EQ(10, (*function1)(&a));
OwnPtr<Function<int(class A*, int)>> unboundFunction2 =
bind<class A*, int>(&A::addF);
EXPECT_EQ(25, (*unboundFunction2)(&a, 15));
OwnPtr<Function<int(int)>> objectBoundFunction2 =
bind<int>(&A::addF, &a);
EXPECT_EQ(25, (*objectBoundFunction2)(15));
}
class Number : public RefCounted<Number> {
public:
static PassRefPtr<Number> create(int value)
{
return adoptRef(new Number(value));
}
~Number()
{
m_value = 0;
}
int value() const { return m_value; }
private:
explicit Number(int value)
: m_value(value)
{
}
int m_value;
};
int multiplyNumberByTwo(Number* number)
{
return number->value() * 2;
}
TEST(FunctionalTest, RefCountedStorage)
{
RefPtr<Number> five = Number::create(5);
OwnPtr<Function<int()>> multiplyFiveByTwoFunction = bind(multiplyNumberByTwo, five);
EXPECT_EQ(10, (*multiplyFiveByTwoFunction)());
OwnPtr<Function<int()>> multiplyFourByTwoFunction = bind(multiplyNumberByTwo, Number::create(4));
EXPECT_EQ(8, (*multiplyFourByTwoFunction)());
RefPtr<Number> six = Number::create(6);
OwnPtr<Function<int()>> multiplySixByTwoFunction = bind(multiplyNumberByTwo, six.release());
EXPECT_FALSE(six);
EXPECT_EQ(12, (*multiplySixByTwoFunction)());
}
int processUnwrappedClass(const UnwrappedClass& u, int v)
{
return u.value() * v;
}
// Tests that:
// - ParamStorageTraits's wrap()/unwrap() are called, and
// - bind()'s arguments are passed as references to ParamStorageTraits::wrap()
// with no additional copies.
// This test would fail in compile if something is wrong,
// rather than in runtime.
TEST(FunctionalTest, WrapUnwrap)
{
OwnPtr<Function<int()>> function = bind(processUnwrappedClass, ClassToBeWrapped(3), 7);
EXPECT_EQ(21, (*function)());
}
TEST(FunctionalTest, WrapUnwrapInPartialBind)
{
OwnPtr<Function<int(int)>> partiallyBoundFunction = bind<int>(processUnwrappedClass, ClassToBeWrapped(3));
EXPECT_EQ(21, (*partiallyBoundFunction)(7));
}
} // anonymous namespace
} // namespace WTF