1
0
mirror of https://github.com/alliedmodders/metamod-source.git synced 2024-12-03 15:24:15 +01:00
HLMetaModOfficial/core/sourcehook/test/testrefret.cpp

320 lines
9.5 KiB
C++

#include "testevents.h"
#include "sourcehook_test.h"
// Tests support for functions which return references
namespace
{
StateList g_States;
SourceHook::ISourceHook *g_SHPtr;
SourceHook::Plugin g_PLID;
MAKE_STATE_1(State_Func1_Pre1, int*); // p1: the ref Func1_Pre1 is going to return
MAKE_STATE_3(State_Func1_Pre2, META_RES, int*, int*); // p1: current status
// p2: override ret
// p3: what this handler is going to supercede with
MAKE_STATE_1(State_Func1, int*); // p1: the ref Func1 is going to return
MAKE_STATE_2(State_Func1_Post1, int*, int*); // p1: orig_ret; p2: override_ret
MAKE_STATE_1(State_Func1_Post2, int*); // p1: what it's going to return
MAKE_STATE_1(State_Func1_Ret, int*); // p1: the ref it returned
MAKE_STATE_2(State_Func2_Pre1, int, const int*); // p1: func's p1; p2: what it's going to ret
MAKE_STATE_2(State_Func2, int, const int*); // p1: func's p1; p2: what it's going to ret
MAKE_STATE_3(State_Func2_Post1, int, const int*, const int*); // p1: func's p1; p2: orig ret; p3: override ret
MAKE_STATE_1(State_Func2_Ret, const int*); // p1: ret
int g_Var;
class Test
{
public:
int m_Var1;
int m_Var2;
Test() : m_Var1(87)
{
}
virtual int& Func1()
{
ADD_STATE(State_Func1(&m_Var1));
return m_Var1;
}
virtual const int& Func2(int p1)
{
ADD_STATE(State_Func2(p1, &m_Var2));
m_Var2 = p1;
return m_Var2;
}
};
SH_DECL_HOOK0(Test, Func1, SH_NOATTRIB, 0, int&);
SH_DECL_MANUALHOOK0(Manual_Test_Func1, 0, 0, 0, int&);
SH_DECL_HOOK1(Test, Func2, SH_NOATTRIB, 0, const int&, int);
SH_DECL_MANUALHOOK1(Manual_Test_Func2, 1, 0, 0, const int&, int);
class CHook
{
public:
int m_Var;
virtual int& Func1_Pre1()
{
ADD_STATE(State_Func1_Pre1(&m_Var));
RETURN_META_VALUE(MRES_OVERRIDE, m_Var);
}
virtual int &Func1_Pre2()
{
int &overrideret = META_RESULT_OVERRIDE_RET(int&);
overrideret = 1337;
ADD_STATE(State_Func1_Pre2(META_RESULT_STATUS, &overrideret, &g_Var));
RETURN_META_VALUE(MRES_SUPERCEDE, g_Var);
}
virtual int& Func1_Post1()
{
ADD_STATE(State_Func1_Post1(&META_RESULT_ORIG_RET(int&), &META_RESULT_OVERRIDE_RET(int&)));
RETURN_META_NOREF(MRES_IGNORED, int&);
}
virtual int& Func1_Post2()
{
ADD_STATE(State_Func1_Post2(&m_Var));
RETURN_META_VALUE(MRES_OVERRIDE, m_Var);
}
virtual const int& Func2_PreRecall_OverrideRet_Normal(int p1)
{
ADD_STATE(State_Func2_Pre1(p1, &m_Var));
RETURN_META_VALUE_NEWPARAMS(MRES_OVERRIDE, m_Var, static_cast<const int& (Test::*)(int)>(&Test::Func2), (1337));
}
virtual const int& Func2_PreRecall_OverrideRet_Manual(int p1)
{
ADD_STATE(State_Func2_Pre1(p1, &m_Var));
RETURN_META_VALUE_MNEWPARAMS(MRES_OVERRIDE, m_Var, Manual_Test_Func2, (1337));
}
virtual const int& Func2_Post1(int p1)
{
ADD_STATE(State_Func2_Post1(p1, &META_RESULT_ORIG_RET(int&), &META_RESULT_OVERRIDE_RET(int&)));
RETURN_META_NOREF(MRES_IGNORED, const int&);
}
};
Test *MyTestFactory()
{
return new Test;
}
}
class TesterRefRet
{
public:
Test *pTest;
CHook hook;
TesterRefRet()
{
pTest = NULL;
}
bool doTests(std::string &error)
{
GET_SHPTR(g_SHPtr);
g_PLID = 1;
pTest = MyTestFactory();
CAutoPtrDestruction<Test> apd(pTest);
int &ret1 = pTest->Func1();
ADD_STATE(State_Func1_Ret(&ret1));
CHECK_STATES((&g_States,
new State_Func1(&pTest->m_Var1),
new State_Func1_Ret(&pTest->m_Var1),
NULL), "Part 1");
// Now add Func1_Pre1, which supercedes and returns hook.m_Var
AddHook__Test_Func1__Func1_Pre();
int &ret2 = pTest->Func1();
ADD_STATE(State_Func1_Ret(&ret2));
CHECK_STATES((&g_States,
new State_Func1_Pre1(&hook.m_Var), // Pre1 says that it's going to override with hook.m_Var
new State_Func1(&pTest->m_Var1), // Function says that it's going to return pTest->m_Var1
new State_Func1_Ret(&hook.m_Var), // hook.m_Var is returned
NULL), "Part 2");
// Now add Func1_Post1, which only reports orig ret and override ret
AddHook__Test_Func1__Func1_Post();
int &ret3 = pTest->Func1();
ADD_STATE(State_Func1_Ret(&ret3));
CHECK_STATES((&g_States,
new State_Func1_Pre1(&hook.m_Var), // Pre1 says that it's going to override with hook.m_Var
new State_Func1(&pTest->m_Var1), // Function says that it's going to return pTest->m_Var1
new State_Func1_Post1(&pTest->m_Var1, &hook.m_Var), // origret(=p1) is what it wanted to
// return, overrideret(=p2) is pre1's var
new State_Func1_Ret(&hook.m_Var), // hook.m_Var is returned
NULL), "Part 3");
// Now add Func1_Pre2, which supercedes and returns g_Var (it also sets the override ret from pre1 to 1337)
// and add Func1_Post2, which overrides and returns hook.m_Var again.
AddHook__Test_Func1__Func1_Pre2();
AddHook__Test_Func1__Func1_Post2();
int &ret4 = pTest->Func1();
ADD_STATE(State_Func1_Ret(&ret4));
CHECK_STATES((&g_States,
new State_Func1_Pre1(&hook.m_Var), // Pre1 says that it's going to override with hook.m_Var
new State_Func1_Pre2(MRES_OVERRIDE, // current status
&hook.m_Var, // override ret (which it set to 1337)
&g_Var), // what it's going to ret, AND supercede with
new State_Func1_Post1(&g_Var, &g_Var), // origret(=p1) is what pre2 superceded with,
// so overrideret(=p2) has to be the same
new State_Func1_Post2(&hook.m_Var), // post2 is going to override with hook.m_Var again
new State_Func1_Ret(&hook.m_Var), // ==>>> hook.m_Var is returned
NULL), "Part 4");
CHECK_COND(hook.m_Var == 1337, "Part 4.1");
// Through a callclass
SourceHook::CallClass<Test> *cc1 = SH_GET_CALLCLASS(pTest);
int &ret5 = SH_CALL(cc1, &Test::Func1)();
ADD_STATE(State_Func1_Ret(&ret5));
CHECK_STATES((&g_States,
new State_Func1(&pTest->m_Var1),
new State_Func1_Ret(&pTest->m_Var1),
NULL), "Part 5");
SH_RELEASE_CALLCLASS(cc1);
////////////////////////////////////////////////////////////////////////////////////////
// Func2 tests
const int &ret21 = pTest->Func2(500);
ADD_STATE(State_Func2_Ret(&ret21));
CHECK_STATES((&g_States,
new State_Func2(500, &pTest->m_Var2),
new State_Func2_Ret(&pTest->m_Var2),
NULL), "Part 6");
AddHook__Test_Func2__Func2_Pre1();
AddHook__Test_Func2__Func2_Post1();
const int &ret22 = pTest->Func2(500);
ADD_STATE(State_Func2_Ret(&ret22));
CHECK_STATES((&g_States,
new State_Func2_Pre1(500, &hook.m_Var), // p1 was 500; it's going to override with hook.m_Var; and also change p1 to 1337
new State_Func2(1337, &pTest->m_Var2), // p1 was 1337; it's going to ret pTest->m_Var2
new State_Func2_Post1(1337, // p1 was 1337
&pTest->m_Var2, // orig ret was pTest->m_Var2
&hook.m_Var), // override ret was hook.m_Var
new State_Func2_Ret(&hook.m_Var), // really returned hook.m_Var
NULL), "Part 7");
return true;
}
virtual void AddHook__Test_Func1__Func1_Pre() = 0;
virtual void AddHook__Test_Func1__Func1_Post() = 0;
virtual void AddHook__Test_Func1__Func1_Pre2() = 0;
virtual void AddHook__Test_Func1__Func1_Post2() = 0;
virtual void AddHook__Test_Func2__Func2_Pre1() = 0;
virtual void AddHook__Test_Func2__Func2_Post1() = 0;
};
class TesterRefRetNonManual : public TesterRefRet
{
public:
void AddHook__Test_Func1__Func1_Pre()
{
SH_ADD_HOOK(Test, Func1, pTest, SH_MEMBER(&hook, &CHook::Func1_Pre1), false);
}
void AddHook__Test_Func1__Func1_Post()
{
SH_ADD_HOOK(Test, Func1, pTest, SH_MEMBER(&hook, &CHook::Func1_Post1), true);
}
void AddHook__Test_Func1__Func1_Pre2()
{
SH_ADD_HOOK(Test, Func1, pTest, SH_MEMBER(&hook, &CHook::Func1_Pre2), false);
}
void AddHook__Test_Func1__Func1_Post2()
{
SH_ADD_HOOK(Test, Func1, pTest, SH_MEMBER(&hook, &CHook::Func1_Post2), true);
}
void AddHook__Test_Func2__Func2_Pre1()
{
SH_ADD_HOOK(Test, Func2, pTest, SH_MEMBER(&hook, &CHook::Func2_PreRecall_OverrideRet_Normal), false);
}
void AddHook__Test_Func2__Func2_Post1()
{
SH_ADD_HOOK(Test, Func2, pTest, SH_MEMBER(&hook, &CHook::Func2_Post1), true);
}
};
class TesterRefRetManual : public TesterRefRet
{
public:
void AddHook__Test_Func1__Func1_Pre()
{
SH_ADD_MANUALHOOK(Manual_Test_Func1, pTest, SH_MEMBER(&hook, &CHook::Func1_Pre1), false);
}
void AddHook__Test_Func1__Func1_Post()
{
SH_ADD_MANUALHOOK(Manual_Test_Func1, pTest, SH_MEMBER(&hook, &CHook::Func1_Post1), true);
}
void AddHook__Test_Func1__Func1_Pre2()
{
SH_ADD_MANUALHOOK(Manual_Test_Func1, pTest, SH_MEMBER(&hook, &CHook::Func1_Pre2), false);
}
void AddHook__Test_Func1__Func1_Post2()
{
SH_ADD_MANUALHOOK(Manual_Test_Func1, pTest, SH_MEMBER(&hook, &CHook::Func1_Post2), true);
}
void AddHook__Test_Func2__Func2_Pre1()
{
SH_ADD_MANUALHOOK(Manual_Test_Func2, pTest, SH_MEMBER(&hook, &CHook::Func2_PreRecall_OverrideRet_Manual), false);
}
void AddHook__Test_Func2__Func2_Post1()
{
SH_ADD_MANUALHOOK(Manual_Test_Func2, pTest, SH_MEMBER(&hook, &CHook::Func2_Post1), true);
}
};
bool TestRefRet(std::string &error)
{
TesterRefRet *testerNonManual = new TesterRefRetNonManual;
bool resNonManual = testerNonManual->doTests(error);
if (!resNonManual)
{
return false;
}
TesterRefRet *testerManual = new TesterRefRetManual;
bool resManual = testerManual->doTests(error);
if (!resManual)
{
return false;
}
return true;
}