195 lines
4.7 KiB
C++
195 lines
4.7 KiB
C++
/*
|
|
* Copyright (C) 2019 ~ 2019 Deepin Technology Co., Ltd.
|
|
*
|
|
* Author: zccrs <zccrs@live.com>
|
|
*
|
|
* Maintainer: zccrs <zhangjide@deepin.com>
|
|
*
|
|
* This program is free software: you can redistribute it and/or modify
|
|
* it under the terms of the GNU Lesser General Public License as published by
|
|
* the Free Software Foundation, either version 3 of the License, or
|
|
* any later version.
|
|
*
|
|
* This program is distributed in the hope that it will be useful,
|
|
* but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
|
* GNU General Public License for more details.
|
|
*
|
|
* You should have received a copy of the GNU Lesser General Public License
|
|
* along with this program. If not, see <http://www.gnu.org/licenses/>.
|
|
*/
|
|
|
|
#include <gtest/gtest.h>
|
|
#include <util/DVtableHook>
|
|
|
|
namespace TestClass {
|
|
class A
|
|
{
|
|
public:
|
|
virtual bool test(int v);
|
|
|
|
virtual ~A()
|
|
{
|
|
}
|
|
};
|
|
bool A::test(int v)
|
|
{
|
|
qDebug() << Q_FUNC_INFO << this << v;
|
|
|
|
return false;
|
|
}
|
|
|
|
class B
|
|
{
|
|
public:
|
|
bool test(int v)
|
|
{
|
|
qDebug() << Q_FUNC_INFO << v;
|
|
|
|
return true;
|
|
}
|
|
};
|
|
} // namespace TestClass
|
|
|
|
class ut_DVtableHook : public testing::Test
|
|
{
|
|
public:
|
|
static void SetUpTestCase()
|
|
{
|
|
qDebug() << "*****************" << __FUNCTION__;
|
|
}
|
|
static void TearDownTestCase()
|
|
{
|
|
qDebug() << "*****************" << __FUNCTION__;
|
|
}
|
|
virtual void SetUp();
|
|
virtual void TearDown();
|
|
virtual ~ut_DVtableHook() {}
|
|
};
|
|
void ut_DVtableHook::SetUp()
|
|
{
|
|
}
|
|
void ut_DVtableHook::TearDown()
|
|
{
|
|
}
|
|
|
|
using namespace TestClass;
|
|
DCORE_USE_NAMESPACE
|
|
|
|
static bool test(A *obj, int v)
|
|
{
|
|
qDebug() << Q_FUNC_INFO << obj << v;
|
|
return true;
|
|
}
|
|
|
|
static bool test2(A *obj, int v, bool v2)
|
|
{
|
|
qDebug() << Q_FUNC_INFO << obj << v << v2;
|
|
return v2;
|
|
}
|
|
|
|
TEST_F(ut_DVtableHook, objectFun2ObjectFun)
|
|
{
|
|
A *a = new A();
|
|
B *b = new B();
|
|
|
|
ASSERT_TRUE(DVtableHook::overrideVfptrFun(a, &A::test, b, &B::test));
|
|
ASSERT_TRUE(DVtableHook::hasVtable(a));
|
|
ASSERT_TRUE(a->test(0));
|
|
DVtableHook::resetVfptrFun(a, &A::test);
|
|
ASSERT_TRUE(!a->test(0));
|
|
delete a;
|
|
ASSERT_TRUE(!DVtableHook::hasVtable(a));
|
|
delete b;
|
|
}
|
|
|
|
TEST_F(ut_DVtableHook, objectFun2Fun)
|
|
{
|
|
A *a = new A();
|
|
ASSERT_TRUE(DVtableHook::overrideVfptrFun(a, &A::test, &test));
|
|
ASSERT_TRUE(a->test(1));
|
|
DVtableHook::resetVtable(a);
|
|
ASSERT_TRUE(!DVtableHook::hasVtable(a));
|
|
|
|
delete a;
|
|
}
|
|
|
|
TEST_F(ut_DVtableHook, objectFun2StdFun)
|
|
{
|
|
A *a = new A();
|
|
ASSERT_TRUE(DVtableHook::overrideVfptrFun(a, &A::test, std::bind(&test2, std::placeholders::_1, std::placeholders::_2, true)));
|
|
ASSERT_TRUE(a->test(2));
|
|
DVtableHook::resetVtable(a);
|
|
ASSERT_TRUE(!DVtableHook::hasVtable(a));
|
|
// not support
|
|
// A *a2 = new A();
|
|
// ASSERT_TRUE(DVtableHook::overrideVfptrFun(a2, &A::test, std::bind(&test2, std::placeholders::_1, std::placeholders::_2, false)));
|
|
// ASSERT_TRUE(!a2->test(2));
|
|
// DVtableHook::resetVtable(a2);
|
|
// ASSERT_TRUE(!DVtableHook::hasVtable(a2));
|
|
delete a;
|
|
}
|
|
|
|
TEST_F(ut_DVtableHook, objectFun2LambdaFun)
|
|
{
|
|
A *a = new A();
|
|
auto lambda1 = [a](A *obj, int v) {
|
|
qDebug() << Q_FUNC_INFO << obj << v;
|
|
return a == obj;
|
|
};
|
|
auto lambda2 = [a](A *obj, int v) {
|
|
qDebug() << Q_FUNC_INFO << obj << v;
|
|
return a != obj;
|
|
};
|
|
ASSERT_TRUE(DVtableHook::overrideVfptrFun(a, &A::test, lambda1));
|
|
ASSERT_TRUE(a->test(3));
|
|
ASSERT_TRUE(DVtableHook::overrideVfptrFun(a, &A::test, lambda2));
|
|
ASSERT_TRUE(!a->test(3));
|
|
DVtableHook::resetVtable(a);
|
|
ASSERT_TRUE(!DVtableHook::hasVtable(a));
|
|
delete a;
|
|
}
|
|
|
|
TEST_F(ut_DVtableHook, fun2ObjectFun)
|
|
{
|
|
B *b = new B();
|
|
ASSERT_TRUE(DVtableHook::overrideVfptrFun(&A::test, b, &B::test));
|
|
A *a = new A();
|
|
ASSERT_TRUE(DVtableHook::getVtableOfObject(a) == DVtableHook::getVtableOfClass<A>());
|
|
ASSERT_TRUE(a->test(4));
|
|
delete a;
|
|
delete b;
|
|
}
|
|
|
|
TEST_F(ut_DVtableHook, fun2Fun)
|
|
{
|
|
ASSERT_TRUE(DVtableHook::overrideVfptrFun(&A::test, &test));
|
|
A *a = new A();
|
|
ASSERT_TRUE(a->test(5));
|
|
delete a;
|
|
}
|
|
|
|
TEST_F(ut_DVtableHook, fun2StdFun)
|
|
{
|
|
A *a = new A();
|
|
ASSERT_TRUE(DVtableHook::overrideVfptrFun(&A::test, std::bind(&test2, std::placeholders::_1, std::placeholders::_2, true)));
|
|
ASSERT_TRUE(a->test(6));
|
|
DVtableHook::resetVtable(a);
|
|
ASSERT_TRUE(!DVtableHook::hasVtable(a));
|
|
delete a;
|
|
}
|
|
|
|
TEST_F(ut_DVtableHook, fun2LambdaFun)
|
|
{
|
|
A *a = new A();
|
|
auto lambda = [a](A *obj, int v) {
|
|
qDebug() << Q_FUNC_INFO << obj << v;
|
|
return a == obj;
|
|
};
|
|
ASSERT_TRUE(DVtableHook::overrideVfptrFun(&A::test, lambda));
|
|
ASSERT_TRUE(a->test(7));
|
|
DVtableHook::resetVtable(a);
|
|
ASSERT_TRUE(!DVtableHook::hasVtable(a));
|
|
delete a;
|
|
}
|