2017-03-06 02:25:46 +05:00
|
|
|
/*
|
|
|
|
* This file is part of the CitizenFX project - http://citizen.re/
|
|
|
|
*
|
|
|
|
* See LICENSE and MENTIONS in the root of the source tree for information
|
|
|
|
* regarding licensing.
|
|
|
|
*/
|
|
|
|
|
|
|
|
#pragma once
|
|
|
|
|
|
|
|
#include <cassert>
|
|
|
|
#include <vector>
|
2017-09-28 02:17:21 +05:00
|
|
|
#include <string_view>
|
2017-03-06 02:25:46 +05:00
|
|
|
|
2017-04-02 22:41:50 +05:00
|
|
|
#pragma warning(push)
|
|
|
|
#pragma warning(disable:4201)
|
|
|
|
|
2017-03-06 02:25:46 +05:00
|
|
|
namespace hook
|
|
|
|
{
|
|
|
|
extern ptrdiff_t baseAddressDifference;
|
|
|
|
|
|
|
|
// sets the base address difference based on an obtained pointer
|
|
|
|
inline void set_base(uintptr_t address)
|
|
|
|
{
|
|
|
|
#ifdef _M_IX86
|
|
|
|
uintptr_t addressDiff = (address - 0x400000);
|
|
|
|
#elif defined(_M_AMD64)
|
|
|
|
uintptr_t addressDiff = (address - 0x140000000);
|
|
|
|
#endif
|
|
|
|
|
|
|
|
// pointer-style cast to ensure unsigned overflow ends up copied directly into a signed value
|
|
|
|
baseAddressDifference = *(ptrdiff_t*)&addressDiff;
|
|
|
|
}
|
|
|
|
|
|
|
|
// sets the base to the process main base
|
|
|
|
void set_base();
|
|
|
|
|
2018-01-21 17:14:02 +05:00
|
|
|
inline uintptr_t getRVA(uintptr_t rva)
|
2017-03-06 02:25:46 +05:00
|
|
|
{
|
|
|
|
set_base();
|
|
|
|
#ifdef _M_IX86
|
2018-01-21 17:14:02 +05:00
|
|
|
return static_cast<uintptr_t>(baseAddressDifference + 0x400000 + rva);
|
2017-03-06 02:25:46 +05:00
|
|
|
#elif defined(_M_AMD64)
|
2018-01-21 17:14:02 +05:00
|
|
|
return static_cast<uintptr_t>(baseAddressDifference(0x140000000 + rva);
|
2017-03-06 02:25:46 +05:00
|
|
|
#endif
|
|
|
|
}
|
|
|
|
|
|
|
|
class pattern_match
|
|
|
|
{
|
|
|
|
private:
|
|
|
|
void* m_pointer;
|
|
|
|
|
|
|
|
public:
|
|
|
|
inline pattern_match(void* pointer)
|
|
|
|
: m_pointer(pointer)
|
|
|
|
{
|
|
|
|
}
|
|
|
|
|
|
|
|
template<typename T>
|
|
|
|
T* get(ptrdiff_t offset = 0) const
|
|
|
|
{
|
|
|
|
char* ptr = reinterpret_cast<char*>(m_pointer);
|
|
|
|
return reinterpret_cast<T*>(ptr + offset);
|
|
|
|
}
|
|
|
|
};
|
|
|
|
|
|
|
|
class pattern
|
|
|
|
{
|
|
|
|
private:
|
|
|
|
std::string m_bytes;
|
|
|
|
std::string m_mask;
|
|
|
|
|
|
|
|
#if PATTERNS_USE_HINTS
|
|
|
|
uint64_t m_hash;
|
|
|
|
#endif
|
|
|
|
|
|
|
|
std::vector<pattern_match> m_matches;
|
|
|
|
|
2017-09-08 03:56:29 +05:00
|
|
|
bool m_matched = false;
|
2017-03-06 02:25:46 +05:00
|
|
|
|
2018-01-21 17:14:02 +05:00
|
|
|
uintptr_t m_rangeStart;
|
|
|
|
uintptr_t m_rangeEnd;
|
2017-03-06 02:25:46 +05:00
|
|
|
|
|
|
|
private:
|
2018-01-21 17:14:02 +05:00
|
|
|
void Initialize(std::string_view pattern);
|
|
|
|
|
2017-09-13 01:53:20 +05:00
|
|
|
bool ConsiderHint(uintptr_t offset);
|
2017-03-06 02:25:46 +05:00
|
|
|
|
|
|
|
void EnsureMatches(uint32_t maxCount);
|
|
|
|
|
2017-04-27 05:18:59 +05:00
|
|
|
inline pattern_match _get_internal(size_t index) const
|
2017-03-06 02:25:46 +05:00
|
|
|
{
|
|
|
|
return m_matches[index];
|
|
|
|
}
|
|
|
|
|
2018-01-21 17:14:02 +05:00
|
|
|
inline pattern(uintptr_t module)
|
|
|
|
: pattern( module, 0 )
|
2017-12-23 22:59:25 +05:00
|
|
|
{
|
|
|
|
}
|
|
|
|
|
|
|
|
inline pattern(uintptr_t begin, uintptr_t end)
|
|
|
|
: m_rangeStart(begin), m_rangeEnd(end)
|
|
|
|
{
|
|
|
|
}
|
|
|
|
|
2017-03-06 02:25:46 +05:00
|
|
|
public:
|
2018-01-14 06:03:58 +05:00
|
|
|
pattern(std::string_view pattern)
|
2018-01-21 17:14:02 +05:00
|
|
|
: pattern(getRVA(0))
|
2017-03-06 02:25:46 +05:00
|
|
|
{
|
2018-01-14 06:03:58 +05:00
|
|
|
Initialize(std::move(pattern));
|
2017-03-06 02:25:46 +05:00
|
|
|
}
|
|
|
|
|
2018-01-14 06:03:58 +05:00
|
|
|
inline pattern(void* module, std::string_view pattern)
|
2018-01-21 17:14:02 +05:00
|
|
|
: pattern(reinterpret_cast<uintptr_t>(module))
|
2017-12-23 22:59:25 +05:00
|
|
|
{
|
2018-01-14 06:03:58 +05:00
|
|
|
Initialize(std::move(pattern));
|
2017-12-23 22:59:25 +05:00
|
|
|
}
|
|
|
|
|
2018-01-14 06:03:58 +05:00
|
|
|
inline pattern(uintptr_t begin, uintptr_t end, std::string_view pattern)
|
2017-12-23 22:59:25 +05:00
|
|
|
: m_rangeStart(begin), m_rangeEnd(end)
|
|
|
|
{
|
2018-01-14 06:03:58 +05:00
|
|
|
Initialize(std::move(pattern));
|
2017-12-23 22:59:25 +05:00
|
|
|
}
|
|
|
|
|
2017-09-08 03:56:29 +05:00
|
|
|
inline pattern&& count(uint32_t expected)
|
2017-03-06 02:25:46 +05:00
|
|
|
{
|
|
|
|
EnsureMatches(expected);
|
|
|
|
assert(m_matches.size() == expected);
|
2017-09-08 03:56:29 +05:00
|
|
|
return std::forward<pattern>(*this);
|
2017-03-06 02:25:46 +05:00
|
|
|
}
|
|
|
|
|
2017-09-08 03:56:29 +05:00
|
|
|
inline pattern&& count_hint(uint32_t expected)
|
2017-03-06 02:25:46 +05:00
|
|
|
{
|
|
|
|
EnsureMatches(expected);
|
2017-09-08 03:56:29 +05:00
|
|
|
return std::forward<pattern>(*this);
|
2017-03-06 02:25:46 +05:00
|
|
|
}
|
|
|
|
|
2017-09-08 03:56:29 +05:00
|
|
|
inline pattern&& clear()
|
2017-03-17 04:48:16 +05:00
|
|
|
{
|
|
|
|
m_matches.clear();
|
|
|
|
m_matched = false;
|
2017-09-08 03:56:29 +05:00
|
|
|
return std::forward<pattern>(*this);
|
2017-04-27 05:18:59 +05:00
|
|
|
}
|
|
|
|
|
2017-03-06 02:25:46 +05:00
|
|
|
inline size_t size()
|
|
|
|
{
|
|
|
|
EnsureMatches(UINT32_MAX);
|
|
|
|
return m_matches.size();
|
|
|
|
}
|
|
|
|
|
|
|
|
inline bool empty()
|
|
|
|
{
|
|
|
|
return size() == 0;
|
|
|
|
}
|
|
|
|
|
2017-04-27 05:18:59 +05:00
|
|
|
inline pattern_match get(size_t index)
|
2017-03-06 02:25:46 +05:00
|
|
|
{
|
|
|
|
EnsureMatches(UINT32_MAX);
|
|
|
|
return _get_internal(index);
|
|
|
|
}
|
|
|
|
|
2017-04-27 05:18:59 +05:00
|
|
|
inline pattern_match get_one()
|
2017-03-06 02:25:46 +05:00
|
|
|
{
|
2017-04-27 05:18:59 +05:00
|
|
|
return std::forward<pattern>(*this).count(1)._get_internal(0);
|
2017-03-06 02:25:46 +05:00
|
|
|
}
|
|
|
|
|
|
|
|
template<typename T = void>
|
|
|
|
inline auto get_first(ptrdiff_t offset = 0)
|
|
|
|
{
|
|
|
|
return get_one().get<T>(offset);
|
|
|
|
}
|
|
|
|
|
2017-05-28 00:44:22 +05:00
|
|
|
template <typename Pred>
|
2018-01-14 06:03:58 +05:00
|
|
|
inline Pred for_each_result(Pred&& pred)
|
2017-05-28 00:44:22 +05:00
|
|
|
{
|
|
|
|
EnsureMatches(UINT32_MAX);
|
|
|
|
for ( auto it : m_matches )
|
|
|
|
{
|
2017-09-08 03:56:29 +05:00
|
|
|
std::forward<Pred>(pred)(it);
|
2017-05-28 00:44:22 +05:00
|
|
|
}
|
2018-01-14 06:03:58 +05:00
|
|
|
return std::forward<Pred>(pred);
|
2017-05-28 00:44:22 +05:00
|
|
|
}
|
|
|
|
|
2017-03-06 02:25:46 +05:00
|
|
|
public:
|
2017-09-13 01:53:20 +05:00
|
|
|
#if PATTERNS_USE_HINTS && PATTERNS_CAN_SERIALIZE_HINTS
|
2017-03-06 02:25:46 +05:00
|
|
|
// define a hint
|
|
|
|
static void hint(uint64_t hash, uintptr_t address);
|
|
|
|
#endif
|
|
|
|
};
|
|
|
|
|
2018-01-14 06:03:58 +05:00
|
|
|
inline pattern make_module_pattern(void* module, std::string_view bytes)
|
2017-03-06 02:25:46 +05:00
|
|
|
{
|
2018-01-14 06:03:58 +05:00
|
|
|
return pattern(module, std::move(bytes));
|
2017-12-23 22:59:25 +05:00
|
|
|
}
|
2017-03-06 02:25:46 +05:00
|
|
|
|
2018-01-14 06:03:58 +05:00
|
|
|
inline pattern make_range_pattern(uintptr_t begin, uintptr_t end, std::string_view bytes)
|
2017-03-17 04:48:16 +05:00
|
|
|
{
|
2018-01-14 06:03:58 +05:00
|
|
|
return pattern(begin, end, std::move(bytes));
|
2017-12-23 22:59:25 +05:00
|
|
|
}
|
2017-03-06 02:25:46 +05:00
|
|
|
|
2018-01-14 06:03:58 +05:00
|
|
|
template<typename T = void>
|
|
|
|
inline auto get_pattern(std::string_view pattern_string, ptrdiff_t offset = 0)
|
2017-03-06 02:25:46 +05:00
|
|
|
{
|
2018-01-14 06:03:58 +05:00
|
|
|
return pattern(std::move(pattern_string)).get_first<T>(offset);
|
2017-03-06 02:25:46 +05:00
|
|
|
}
|
2017-04-02 22:41:50 +05:00
|
|
|
}
|
|
|
|
|
|
|
|
#pragma warning(pop)
|