2014-07-31 18:08:02 +02:00
|
|
|
#pragma once
|
|
|
|
|
|
2014-11-10 01:21:50 +01:00
|
|
|
class CPUThread;
|
2014-09-12 15:08:24 +02:00
|
|
|
|
2014-07-31 18:08:02 +02:00
|
|
|
namespace vm
|
|
|
|
|
{
|
2014-08-01 18:27:48 +02:00
|
|
|
template<typename T, int lvl = 1, typename AT = u32>
|
2014-09-01 18:16:44 +02:00
|
|
|
class _ptr_base
|
2014-08-01 18:27:48 +02:00
|
|
|
{
|
|
|
|
|
AT m_addr;
|
|
|
|
|
|
|
|
|
|
public:
|
2014-11-19 15:16:30 +01:00
|
|
|
typedef typename std::remove_cv<T>::type type;
|
2015-01-19 22:30:33 +01:00
|
|
|
static const u32 address_size = sizeof(AT);
|
2014-09-01 18:16:44 +02:00
|
|
|
|
|
|
|
|
_ptr_base operator++ (int)
|
2014-08-01 18:27:48 +02:00
|
|
|
{
|
|
|
|
|
AT result = m_addr;
|
2015-01-14 14:57:19 +01:00
|
|
|
m_addr += address_size;
|
2014-09-01 18:16:44 +02:00
|
|
|
return make(result);
|
2014-08-01 18:27:48 +02:00
|
|
|
}
|
|
|
|
|
|
2014-09-01 18:16:44 +02:00
|
|
|
_ptr_base& operator++ ()
|
2014-08-01 18:27:48 +02:00
|
|
|
{
|
2015-01-14 14:57:19 +01:00
|
|
|
m_addr += address_size;
|
2014-08-01 18:27:48 +02:00
|
|
|
return *this;
|
|
|
|
|
}
|
|
|
|
|
|
2014-09-01 18:16:44 +02:00
|
|
|
_ptr_base operator-- (int)
|
2014-08-01 18:27:48 +02:00
|
|
|
{
|
|
|
|
|
AT result = m_addr;
|
2015-01-14 14:57:19 +01:00
|
|
|
m_addr -= address_size;
|
2014-09-01 18:16:44 +02:00
|
|
|
return make(result);
|
2014-08-01 18:27:48 +02:00
|
|
|
}
|
|
|
|
|
|
2014-09-01 18:16:44 +02:00
|
|
|
_ptr_base& operator-- ()
|
2014-08-01 18:27:48 +02:00
|
|
|
{
|
2015-01-14 14:57:19 +01:00
|
|
|
m_addr -= address_size;
|
2014-08-01 18:27:48 +02:00
|
|
|
return *this;
|
|
|
|
|
}
|
|
|
|
|
|
2014-11-19 15:16:30 +01:00
|
|
|
_ptr_base& operator += (AT count)
|
2014-08-01 18:27:48 +02:00
|
|
|
{
|
2015-01-14 14:57:19 +01:00
|
|
|
m_addr += count * address_size;
|
2014-08-01 18:27:48 +02:00
|
|
|
return *this;
|
|
|
|
|
}
|
|
|
|
|
|
2014-11-19 15:16:30 +01:00
|
|
|
_ptr_base& operator -= (AT count)
|
2014-08-01 18:27:48 +02:00
|
|
|
{
|
2015-01-14 14:57:19 +01:00
|
|
|
m_addr -= count * address_size;
|
2014-08-01 18:27:48 +02:00
|
|
|
return *this;
|
|
|
|
|
}
|
|
|
|
|
|
2015-01-14 14:57:19 +01:00
|
|
|
_ptr_base operator + (typename remove_be_t<AT>::type count) const { return make(m_addr + count * address_size); }
|
|
|
|
|
_ptr_base operator + (typename to_be_t<AT>::type count) const { return make(m_addr + count * address_size); }
|
|
|
|
|
_ptr_base operator - (typename remove_be_t<AT>::type count) const { return make(m_addr - count * address_size); }
|
|
|
|
|
_ptr_base operator - (typename to_be_t<AT>::type count) const { return make(m_addr - count * address_size); }
|
2014-08-01 18:27:48 +02:00
|
|
|
|
2014-11-19 15:16:30 +01:00
|
|
|
__forceinline bool operator <(const _ptr_base& right) const { return m_addr < right.m_addr; }
|
|
|
|
|
__forceinline bool operator <=(const _ptr_base& right) const { return m_addr <= right.m_addr; }
|
|
|
|
|
__forceinline bool operator >(const _ptr_base& right) const { return m_addr > right.m_addr; }
|
|
|
|
|
__forceinline bool operator >=(const _ptr_base& right) const { return m_addr >= right.m_addr; }
|
|
|
|
|
__forceinline bool operator ==(const _ptr_base& right) const { return m_addr == right.m_addr; }
|
|
|
|
|
__forceinline bool operator !=(const _ptr_base& right) const { return m_addr != right.m_addr; }
|
|
|
|
|
__forceinline bool operator ==(const nullptr_t& right) const { return m_addr == 0; }
|
|
|
|
|
__forceinline bool operator !=(const nullptr_t& right) const { return m_addr != 0; }
|
|
|
|
|
explicit operator bool() const { return m_addr != 0; }
|
2014-08-01 18:27:48 +02:00
|
|
|
|
2014-10-01 11:45:43 +02:00
|
|
|
__forceinline _ptr_base<T, lvl - 1, std::conditional<is_be_t<T>::value, typename to_be_t<AT>::type, AT>>& operator *() const
|
2014-08-01 18:27:48 +02:00
|
|
|
{
|
2015-01-14 14:57:19 +01:00
|
|
|
return vm::get_ref<_ptr_base<T, lvl - 1, std::conditional<is_be_t<T>::value, typename to_be_t<AT>::type, AT>>>(vm::cast(m_addr));
|
2014-08-01 18:27:48 +02:00
|
|
|
}
|
|
|
|
|
|
2014-11-19 15:16:30 +01:00
|
|
|
__forceinline _ptr_base<T, lvl - 1, std::conditional<is_be_t<T>::value, typename to_be_t<AT>::type, AT>>& operator [](AT index) const
|
2014-08-01 18:27:48 +02:00
|
|
|
{
|
2015-01-14 14:57:19 +01:00
|
|
|
return vm::get_ref<_ptr_base<T, lvl - 1, std::conditional<is_be_t<T>::value, typename to_be_t<AT>::type, AT>>>(vm::cast(m_addr + sizeof(AT)* index));
|
2014-08-01 18:27:48 +02:00
|
|
|
}
|
|
|
|
|
|
2014-09-04 19:32:20 +02:00
|
|
|
template<typename AT2>
|
|
|
|
|
operator const _ptr_base<T, lvl, AT2>() const
|
2014-08-01 18:27:48 +02:00
|
|
|
{
|
2015-01-19 22:30:33 +01:00
|
|
|
const AT2 addr = convert_le_be<AT2>(m_addr);
|
|
|
|
|
return reinterpret_cast<const _ptr_base<T, lvl, AT2>&>(addr);
|
2014-08-01 18:27:48 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
AT addr() const
|
|
|
|
|
{
|
|
|
|
|
return m_addr;
|
|
|
|
|
}
|
|
|
|
|
|
2014-09-11 21:18:19 +02:00
|
|
|
void set(const AT value)
|
|
|
|
|
{
|
|
|
|
|
m_addr = value;
|
|
|
|
|
}
|
|
|
|
|
|
2015-01-19 22:30:33 +01:00
|
|
|
static const _ptr_base make(const AT& addr)
|
2014-08-01 18:27:48 +02:00
|
|
|
{
|
2015-01-19 22:30:33 +01:00
|
|
|
return reinterpret_cast<const _ptr_base&>(addr);
|
2014-08-01 18:27:48 +02:00
|
|
|
}
|
2014-09-04 19:32:20 +02:00
|
|
|
|
|
|
|
|
_ptr_base& operator = (const _ptr_base& right) = default;
|
2014-08-01 18:27:48 +02:00
|
|
|
};
|
|
|
|
|
|
|
|
|
|
template<typename T, typename AT>
|
2014-09-01 18:16:44 +02:00
|
|
|
class _ptr_base<T, 1, AT>
|
2014-08-01 18:27:48 +02:00
|
|
|
{
|
|
|
|
|
AT m_addr;
|
|
|
|
|
|
|
|
|
|
public:
|
2014-11-19 15:16:30 +01:00
|
|
|
typedef typename std::remove_cv<T>::type type;
|
2015-01-19 22:30:33 +01:00
|
|
|
static const u32 data_size = sizeof(T);
|
2014-11-19 15:16:30 +01:00
|
|
|
|
2014-09-04 19:32:20 +02:00
|
|
|
__forceinline T* const operator -> () const
|
2014-08-01 18:27:48 +02:00
|
|
|
{
|
2015-01-14 14:57:19 +01:00
|
|
|
return vm::get_ptr<T>(vm::cast(m_addr));
|
2014-08-01 18:27:48 +02:00
|
|
|
}
|
|
|
|
|
|
2014-09-01 18:16:44 +02:00
|
|
|
_ptr_base operator++ (int)
|
2014-08-01 18:27:48 +02:00
|
|
|
{
|
|
|
|
|
AT result = m_addr;
|
2015-01-14 14:57:19 +01:00
|
|
|
m_addr += data_size;
|
2014-09-01 18:16:44 +02:00
|
|
|
return make(result);
|
2014-08-01 18:27:48 +02:00
|
|
|
}
|
|
|
|
|
|
2014-09-01 18:16:44 +02:00
|
|
|
_ptr_base& operator++ ()
|
2014-08-01 18:27:48 +02:00
|
|
|
{
|
2015-01-14 14:57:19 +01:00
|
|
|
m_addr += data_size;
|
2014-08-01 18:27:48 +02:00
|
|
|
return *this;
|
|
|
|
|
}
|
|
|
|
|
|
2014-09-01 18:16:44 +02:00
|
|
|
_ptr_base operator-- (int)
|
2014-08-01 18:27:48 +02:00
|
|
|
{
|
|
|
|
|
AT result = m_addr;
|
2015-01-14 14:57:19 +01:00
|
|
|
m_addr -= data_size;
|
2014-09-01 18:16:44 +02:00
|
|
|
return make(result);
|
2014-08-01 18:27:48 +02:00
|
|
|
}
|
|
|
|
|
|
2014-09-01 18:16:44 +02:00
|
|
|
_ptr_base& operator-- ()
|
2014-08-01 18:27:48 +02:00
|
|
|
{
|
2015-01-14 14:57:19 +01:00
|
|
|
m_addr -= data_size;
|
2014-08-01 18:27:48 +02:00
|
|
|
return *this;
|
|
|
|
|
}
|
|
|
|
|
|
2014-11-19 15:16:30 +01:00
|
|
|
_ptr_base& operator += (AT count)
|
2014-08-01 18:27:48 +02:00
|
|
|
{
|
2015-01-14 14:57:19 +01:00
|
|
|
m_addr += count * data_size;
|
2014-08-01 18:27:48 +02:00
|
|
|
return *this;
|
|
|
|
|
}
|
|
|
|
|
|
2014-11-19 15:16:30 +01:00
|
|
|
_ptr_base& operator -= (AT count)
|
2014-08-01 18:27:48 +02:00
|
|
|
{
|
2015-01-14 14:57:19 +01:00
|
|
|
m_addr -= count * data_size;
|
2014-08-01 18:27:48 +02:00
|
|
|
return *this;
|
|
|
|
|
}
|
|
|
|
|
|
2015-01-14 14:57:19 +01:00
|
|
|
_ptr_base operator + (typename remove_be_t<AT>::type count) const { return make(m_addr + count * data_size); }
|
|
|
|
|
_ptr_base operator + (typename to_be_t<AT>::type count) const { return make(m_addr + count * data_size); }
|
|
|
|
|
_ptr_base operator - (typename remove_be_t<AT>::type count) const { return make(m_addr - count * data_size); }
|
|
|
|
|
_ptr_base operator - (typename to_be_t<AT>::type count) const { return make(m_addr - count * data_size); }
|
2014-08-01 18:27:48 +02:00
|
|
|
|
2014-09-04 19:32:20 +02:00
|
|
|
__forceinline T& operator *() const
|
2014-08-01 18:27:48 +02:00
|
|
|
{
|
2015-01-14 14:57:19 +01:00
|
|
|
return vm::get_ref<T>(vm::cast(m_addr));
|
2014-08-01 18:27:48 +02:00
|
|
|
}
|
|
|
|
|
|
2014-11-20 20:41:04 +01:00
|
|
|
__forceinline T& operator [](typename remove_be_t<AT>::type index) const
|
2014-08-01 18:27:48 +02:00
|
|
|
{
|
2015-01-14 14:57:19 +01:00
|
|
|
return vm::get_ref<T>(vm::cast(m_addr + data_size * index));
|
2014-08-01 18:27:48 +02:00
|
|
|
}
|
|
|
|
|
|
2014-11-20 20:41:04 +01:00
|
|
|
__forceinline T& operator [](typename to_be_t<AT>::forced_type index) const
|
2014-08-01 18:27:48 +02:00
|
|
|
{
|
2015-01-14 14:57:19 +01:00
|
|
|
return vm::get_ref<T>(vm::cast(m_addr + data_size * index));
|
2014-08-01 18:27:48 +02:00
|
|
|
}
|
2014-11-19 15:16:30 +01:00
|
|
|
|
|
|
|
|
__forceinline bool operator <(const _ptr_base& right) const { return m_addr < right.m_addr; }
|
|
|
|
|
__forceinline bool operator <=(const _ptr_base& right) const { return m_addr <= right.m_addr; }
|
|
|
|
|
__forceinline bool operator >(const _ptr_base& right) const { return m_addr > right.m_addr; }
|
|
|
|
|
__forceinline bool operator >=(const _ptr_base& right) const { return m_addr >= right.m_addr; }
|
|
|
|
|
__forceinline bool operator ==(const _ptr_base& right) const { return m_addr == right.m_addr; }
|
|
|
|
|
__forceinline bool operator !=(const _ptr_base& right) const { return m_addr != right.m_addr; }
|
|
|
|
|
__forceinline bool operator ==(const nullptr_t& right) const { return m_addr == 0; }
|
|
|
|
|
__forceinline bool operator !=(const nullptr_t& right) const { return m_addr != 0; }
|
|
|
|
|
explicit operator bool() const { return m_addr != 0; }
|
|
|
|
|
explicit operator T*() const { return get_ptr(); }
|
2014-08-01 18:27:48 +02:00
|
|
|
|
|
|
|
|
AT addr() const
|
|
|
|
|
{
|
|
|
|
|
return m_addr;
|
|
|
|
|
}
|
|
|
|
|
|
2014-11-21 14:52:01 +01:00
|
|
|
template<typename U>
|
|
|
|
|
void set(U&& value)
|
2014-09-11 21:18:19 +02:00
|
|
|
{
|
2014-11-20 20:41:04 +01:00
|
|
|
m_addr = convert_le_be<AT>(value);
|
2014-09-11 21:18:19 +02:00
|
|
|
}
|
|
|
|
|
|
2014-09-04 19:32:20 +02:00
|
|
|
template<typename AT2>
|
|
|
|
|
operator const _ptr_base<T, 1, AT2>() const
|
|
|
|
|
{
|
2015-01-19 22:30:33 +01:00
|
|
|
const AT2 addr = convert_le_be<AT2>(m_addr);
|
|
|
|
|
return reinterpret_cast<const _ptr_base<T, 1, AT2>&>(addr);
|
2014-09-04 19:32:20 +02:00
|
|
|
}
|
|
|
|
|
|
2014-11-19 15:16:30 +01:00
|
|
|
T* get_ptr() const
|
2014-08-01 18:27:48 +02:00
|
|
|
{
|
2015-01-14 14:57:19 +01:00
|
|
|
return vm::get_ptr<T>(vm::cast(m_addr));
|
2014-08-01 18:27:48 +02:00
|
|
|
}
|
|
|
|
|
|
2015-01-19 22:30:33 +01:00
|
|
|
static const _ptr_base make(const AT& addr)
|
2014-08-01 18:27:48 +02:00
|
|
|
{
|
2015-01-19 22:30:33 +01:00
|
|
|
return reinterpret_cast<const _ptr_base&>(addr);
|
2014-08-01 18:27:48 +02:00
|
|
|
}
|
2014-09-04 19:32:20 +02:00
|
|
|
|
|
|
|
|
_ptr_base& operator = (const _ptr_base& right) = default;
|
2014-08-01 18:27:48 +02:00
|
|
|
};
|
|
|
|
|
|
|
|
|
|
template<typename AT>
|
2014-09-01 18:16:44 +02:00
|
|
|
class _ptr_base<void, 1, AT>
|
2014-08-01 18:27:48 +02:00
|
|
|
{
|
|
|
|
|
AT m_addr;
|
|
|
|
|
|
|
|
|
|
public:
|
|
|
|
|
AT addr() const
|
|
|
|
|
{
|
|
|
|
|
return m_addr;
|
|
|
|
|
}
|
|
|
|
|
|
2014-09-11 21:18:19 +02:00
|
|
|
void set(const AT value)
|
|
|
|
|
{
|
|
|
|
|
m_addr = value;
|
|
|
|
|
}
|
|
|
|
|
|
2014-11-19 15:16:30 +01:00
|
|
|
void* get_ptr() const
|
2014-08-01 18:27:48 +02:00
|
|
|
{
|
2015-01-14 14:57:19 +01:00
|
|
|
return vm::get_ptr<void>(vm::cast(m_addr));
|
2014-08-01 18:27:48 +02:00
|
|
|
}
|
|
|
|
|
|
2014-11-19 15:16:30 +01:00
|
|
|
explicit operator void*() const
|
2014-08-01 18:27:48 +02:00
|
|
|
{
|
2014-11-19 15:16:30 +01:00
|
|
|
return get_ptr();
|
2014-08-01 18:27:48 +02:00
|
|
|
}
|
|
|
|
|
|
2014-11-19 15:16:30 +01:00
|
|
|
__forceinline bool operator <(const _ptr_base& right) const { return m_addr < right.m_addr; }
|
|
|
|
|
__forceinline bool operator <=(const _ptr_base& right) const { return m_addr <= right.m_addr; }
|
|
|
|
|
__forceinline bool operator >(const _ptr_base& right) const { return m_addr > right.m_addr; }
|
|
|
|
|
__forceinline bool operator >=(const _ptr_base& right) const { return m_addr >= right.m_addr; }
|
|
|
|
|
__forceinline bool operator ==(const _ptr_base& right) const { return m_addr == right.m_addr; }
|
|
|
|
|
__forceinline bool operator !=(const _ptr_base& right) const { return m_addr != right.m_addr; }
|
|
|
|
|
__forceinline bool operator ==(const nullptr_t& right) const { return m_addr == 0; }
|
|
|
|
|
__forceinline bool operator !=(const nullptr_t& right) const { return m_addr != 0; }
|
|
|
|
|
explicit operator bool() const { return m_addr != 0; }
|
|
|
|
|
|
2014-09-04 19:32:20 +02:00
|
|
|
template<typename AT2>
|
|
|
|
|
operator const _ptr_base<void, 1, AT2>() const
|
|
|
|
|
{
|
2015-01-19 22:30:33 +01:00
|
|
|
const AT2 addr = convert_le_be<AT2>(m_addr);
|
|
|
|
|
return reinterpret_cast<const _ptr_base<void, 1, AT2>&>(addr);
|
2014-09-04 19:32:20 +02:00
|
|
|
}
|
|
|
|
|
|
2014-09-08 02:54:17 +02:00
|
|
|
template<typename AT2>
|
|
|
|
|
operator const _ptr_base<const void, 1, AT2>() const
|
|
|
|
|
{
|
2015-01-19 22:30:33 +01:00
|
|
|
const AT2 addr = convert_le_be<AT2>(m_addr);
|
|
|
|
|
return reinterpret_cast<const _ptr_base<const void, 1, AT2>&>(addr);
|
2014-09-08 02:54:17 +02:00
|
|
|
}
|
|
|
|
|
|
2015-01-19 22:30:33 +01:00
|
|
|
static const _ptr_base make(const AT& addr)
|
2014-08-31 17:01:48 +02:00
|
|
|
{
|
2015-01-19 22:30:33 +01:00
|
|
|
return reinterpret_cast<const _ptr_base&>(addr);
|
2014-08-31 17:01:48 +02:00
|
|
|
}
|
2014-09-04 19:32:20 +02:00
|
|
|
|
|
|
|
|
_ptr_base& operator = (const _ptr_base& right) = default;
|
2014-08-31 17:01:48 +02:00
|
|
|
};
|
|
|
|
|
|
|
|
|
|
template<typename AT>
|
2014-09-01 18:16:44 +02:00
|
|
|
class _ptr_base<const void, 1, AT>
|
2014-08-31 17:01:48 +02:00
|
|
|
{
|
|
|
|
|
AT m_addr;
|
|
|
|
|
|
|
|
|
|
public:
|
|
|
|
|
AT addr() const
|
|
|
|
|
{
|
|
|
|
|
return m_addr;
|
|
|
|
|
}
|
|
|
|
|
|
2014-09-11 21:18:19 +02:00
|
|
|
void set(const AT value)
|
|
|
|
|
{
|
|
|
|
|
m_addr = value;
|
|
|
|
|
}
|
|
|
|
|
|
2014-11-19 15:16:30 +01:00
|
|
|
const void* get_ptr() const
|
2014-08-31 17:01:48 +02:00
|
|
|
{
|
2015-01-14 14:57:19 +01:00
|
|
|
return vm::get_ptr<const void>(vm::cast(m_addr));
|
2014-08-31 17:01:48 +02:00
|
|
|
}
|
|
|
|
|
|
2014-11-19 15:16:30 +01:00
|
|
|
explicit operator const void*() const
|
2014-08-31 17:01:48 +02:00
|
|
|
{
|
2014-11-19 15:16:30 +01:00
|
|
|
return get_ptr();
|
2014-08-31 17:01:48 +02:00
|
|
|
}
|
|
|
|
|
|
2014-11-19 15:16:30 +01:00
|
|
|
__forceinline bool operator <(const _ptr_base& right) const { return m_addr < right.m_addr; }
|
|
|
|
|
__forceinline bool operator <=(const _ptr_base& right) const { return m_addr <= right.m_addr; }
|
|
|
|
|
__forceinline bool operator >(const _ptr_base& right) const { return m_addr > right.m_addr; }
|
|
|
|
|
__forceinline bool operator >=(const _ptr_base& right) const { return m_addr >= right.m_addr; }
|
|
|
|
|
__forceinline bool operator ==(const _ptr_base& right) const { return m_addr == right.m_addr; }
|
|
|
|
|
__forceinline bool operator !=(const _ptr_base& right) const { return m_addr != right.m_addr; }
|
|
|
|
|
__forceinline bool operator ==(const nullptr_t& right) const { return m_addr == 0; }
|
|
|
|
|
__forceinline bool operator !=(const nullptr_t& right) const { return m_addr != 0; }
|
|
|
|
|
explicit operator bool() const { return m_addr != 0; }
|
|
|
|
|
|
2014-09-04 19:32:20 +02:00
|
|
|
template<typename AT2>
|
|
|
|
|
operator const _ptr_base<const void, 1, AT2>() const
|
|
|
|
|
{
|
2015-01-19 22:30:33 +01:00
|
|
|
const AT2 addr = convert_le_be<AT2>(m_addr);
|
|
|
|
|
return reinterpret_cast<const _ptr_base<const void, 1, AT2>&>(addr);
|
2014-09-04 19:32:20 +02:00
|
|
|
}
|
|
|
|
|
|
2015-01-19 22:30:33 +01:00
|
|
|
static const _ptr_base make(const AT& addr)
|
2014-08-01 18:27:48 +02:00
|
|
|
{
|
2015-01-19 22:30:33 +01:00
|
|
|
return reinterpret_cast<const _ptr_base&>(addr);
|
2014-08-01 18:27:48 +02:00
|
|
|
}
|
2014-09-04 19:32:20 +02:00
|
|
|
|
|
|
|
|
_ptr_base& operator = (const _ptr_base& right) = default;
|
2014-08-01 18:27:48 +02:00
|
|
|
};
|
|
|
|
|
|
|
|
|
|
template<typename AT, typename RT, typename ...T>
|
2014-09-01 18:16:44 +02:00
|
|
|
class _ptr_base<RT(*)(T...), 1, AT>
|
2014-08-01 18:27:48 +02:00
|
|
|
{
|
|
|
|
|
AT m_addr;
|
|
|
|
|
|
|
|
|
|
public:
|
|
|
|
|
typedef RT(*type)(T...);
|
|
|
|
|
|
2015-01-11 11:43:40 +01:00
|
|
|
RT operator()(CPUThread& CPU, T... args) const; // defined in CB_FUNC.h, call using specified CPU thread context
|
2014-09-12 15:08:24 +02:00
|
|
|
|
2015-01-02 00:41:29 +01:00
|
|
|
RT operator()(T... args) const; // defined in CB_FUNC.h, call using current CPU thread context
|
2014-08-01 18:27:48 +02:00
|
|
|
|
2014-09-11 21:18:19 +02:00
|
|
|
AT addr() const
|
2014-08-01 18:27:48 +02:00
|
|
|
{
|
2014-09-11 21:18:19 +02:00
|
|
|
return m_addr;
|
2014-08-01 18:27:48 +02:00
|
|
|
}
|
|
|
|
|
|
2014-09-11 21:18:19 +02:00
|
|
|
void set(const AT value)
|
2014-08-01 18:27:48 +02:00
|
|
|
{
|
2014-09-11 21:18:19 +02:00
|
|
|
m_addr = value;
|
2014-08-01 18:27:48 +02:00
|
|
|
}
|
|
|
|
|
|
2014-11-19 15:16:30 +01:00
|
|
|
__forceinline bool operator <(const _ptr_base& right) const { return m_addr < right.m_addr; }
|
|
|
|
|
__forceinline bool operator <=(const _ptr_base& right) const { return m_addr <= right.m_addr; }
|
|
|
|
|
__forceinline bool operator >(const _ptr_base& right) const { return m_addr > right.m_addr; }
|
|
|
|
|
__forceinline bool operator >=(const _ptr_base& right) const { return m_addr >= right.m_addr; }
|
|
|
|
|
__forceinline bool operator ==(const _ptr_base& right) const { return m_addr == right.m_addr; }
|
|
|
|
|
__forceinline bool operator !=(const _ptr_base& right) const { return m_addr != right.m_addr; }
|
|
|
|
|
__forceinline bool operator ==(const nullptr_t& right) const { return m_addr == 0; }
|
|
|
|
|
__forceinline bool operator !=(const nullptr_t& right) const { return m_addr != 0; }
|
|
|
|
|
explicit operator bool() const { return m_addr != 0; }
|
2014-08-01 18:27:48 +02:00
|
|
|
|
2014-09-04 19:32:20 +02:00
|
|
|
template<typename AT2>
|
|
|
|
|
operator const _ptr_base<RT(*)(T...), 1, AT2>() const
|
|
|
|
|
{
|
2015-01-19 22:30:33 +01:00
|
|
|
const AT2 addr = convert_le_be<AT2>(m_addr);
|
|
|
|
|
return reinterpret_cast<const _ptr_base<RT(*)(T...), 1, AT2>&>(addr);
|
2014-09-04 19:32:20 +02:00
|
|
|
}
|
|
|
|
|
|
2015-01-19 22:30:33 +01:00
|
|
|
static const _ptr_base make(const AT& addr)
|
2014-08-01 18:27:48 +02:00
|
|
|
{
|
2015-01-19 22:30:33 +01:00
|
|
|
return reinterpret_cast<const _ptr_base&>(addr);
|
2014-08-01 18:27:48 +02:00
|
|
|
}
|
2014-09-01 14:47:26 +02:00
|
|
|
|
2014-09-04 19:32:20 +02:00
|
|
|
operator const std::function<RT(T...)>() const
|
2014-09-01 14:47:26 +02:00
|
|
|
{
|
2015-01-19 22:30:33 +01:00
|
|
|
const AT addr = convert_le_be<AT>(m_addr);
|
2014-09-01 14:47:26 +02:00
|
|
|
return [addr](T... args) -> RT { return make(addr)(args...); };
|
|
|
|
|
}
|
2014-09-04 19:32:20 +02:00
|
|
|
|
|
|
|
|
_ptr_base& operator = (const _ptr_base& right) = default;
|
2014-08-01 18:27:48 +02:00
|
|
|
};
|
|
|
|
|
|
2014-09-01 18:16:44 +02:00
|
|
|
//BE pointer to LE data
|
2014-09-02 00:22:13 +02:00
|
|
|
template<typename T, int lvl = 1, typename AT = u32> struct bptrl : public _ptr_base<T, lvl, typename to_be_t<AT>::type>
|
|
|
|
|
{
|
2015-01-19 22:30:33 +01:00
|
|
|
static const bptrl make(AT addr)
|
2014-09-02 00:22:13 +02:00
|
|
|
{
|
2015-01-19 17:59:07 +01:00
|
|
|
auto res = _ptr_base<T, lvl, typename to_be_t<AT>::type>::make(convert_le_be<typename to_be_t<AT>::type>(addr));
|
2015-01-19 22:30:33 +01:00
|
|
|
return static_cast<const bptrl&>(res);
|
2014-09-02 00:22:13 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
using _ptr_base<T, lvl, typename to_be_t<AT>::type>::operator=;
|
|
|
|
|
};
|
|
|
|
|
|
2014-09-01 18:16:44 +02:00
|
|
|
//BE pointer to BE data
|
2014-09-02 00:22:13 +02:00
|
|
|
template<typename T, int lvl = 1, typename AT = u32> struct bptrb : public _ptr_base<typename to_be_t<T>::type, lvl, typename to_be_t<AT>::type>
|
|
|
|
|
{
|
2015-01-19 22:30:33 +01:00
|
|
|
static const bptrb make(AT addr)
|
2014-09-02 00:22:13 +02:00
|
|
|
{
|
2015-01-19 17:59:07 +01:00
|
|
|
auto res = _ptr_base<typename to_be_t<T>::type, lvl, typename to_be_t<AT>::type>::make(convert_le_be<typename to_be_t<AT>::type>(addr));
|
2015-01-19 22:30:33 +01:00
|
|
|
return static_cast<const bptrb&>(res);
|
2014-09-02 00:22:13 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
using _ptr_base<typename to_be_t<T>::type, lvl, typename to_be_t<AT>::type>::operator=;
|
|
|
|
|
};
|
|
|
|
|
|
2014-09-01 18:16:44 +02:00
|
|
|
//LE pointer to BE data
|
2014-09-02 00:22:13 +02:00
|
|
|
template<typename T, int lvl = 1, typename AT = u32> struct lptrb : public _ptr_base<typename to_be_t<T>::type, lvl, AT>
|
|
|
|
|
{
|
2015-01-19 22:30:33 +01:00
|
|
|
static const lptrb make(AT addr)
|
2014-09-02 00:22:13 +02:00
|
|
|
{
|
2015-01-19 17:59:07 +01:00
|
|
|
auto res = _ptr_base<typename to_be_t<T>::type, lvl, AT>::make(addr);
|
2015-01-19 22:30:33 +01:00
|
|
|
return static_cast<const lptrb&>(res);
|
2014-09-02 00:22:13 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
using _ptr_base<typename to_be_t<T>::type, lvl, AT>::operator=;
|
|
|
|
|
};
|
|
|
|
|
|
2014-09-01 18:16:44 +02:00
|
|
|
//LE pointer to LE data
|
2014-09-02 00:22:13 +02:00
|
|
|
template<typename T, int lvl = 1, typename AT = u32> struct lptrl : public _ptr_base<T, lvl, AT>
|
|
|
|
|
{
|
2015-01-19 22:30:33 +01:00
|
|
|
static const lptrl make(AT addr)
|
2014-09-02 00:22:13 +02:00
|
|
|
{
|
2015-01-19 17:59:07 +01:00
|
|
|
auto res = _ptr_base<T, lvl, AT>::make(addr);
|
2015-01-19 22:30:33 +01:00
|
|
|
return static_cast<const lptrl&>(res);
|
2014-09-02 00:22:13 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
using _ptr_base<T, lvl, AT>::operator=;
|
|
|
|
|
};
|
|
|
|
|
|
2014-09-01 18:16:44 +02:00
|
|
|
namespace ps3
|
|
|
|
|
{
|
2015-01-19 17:30:35 +01:00
|
|
|
template<typename T, int lvl = 1, typename AT = u32> struct ptr;
|
|
|
|
|
template<typename T, int lvl = 1, typename AT = u32> struct bptr;
|
|
|
|
|
|
2014-11-19 15:16:30 +01:00
|
|
|
//default pointer for HLE functions (LE pointer to BE data)
|
2015-01-19 17:30:35 +01:00
|
|
|
template<typename T, int lvl, typename AT> struct ptr : public lptrb<T, lvl, AT>
|
2014-09-01 18:16:44 +02:00
|
|
|
{
|
2015-01-19 22:30:33 +01:00
|
|
|
static const ptr make(AT addr)
|
2014-09-01 18:16:44 +02:00
|
|
|
{
|
2015-01-19 17:59:07 +01:00
|
|
|
auto res = lptrb<T, lvl, AT>::make(addr);
|
2015-01-19 22:30:33 +01:00
|
|
|
return static_cast<const ptr&>(res);
|
2015-01-19 17:30:35 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
vm::ps3::bptr<T, lvl, AT> to_be() const
|
|
|
|
|
{
|
2015-01-19 17:59:07 +01:00
|
|
|
return vm::ps3::bptr<T, lvl, AT>::make(this->addr());
|
2014-09-01 18:16:44 +02:00
|
|
|
}
|
2014-09-02 00:22:13 +02:00
|
|
|
|
|
|
|
|
using lptrb<T, lvl, AT>::operator=;
|
2014-09-01 18:16:44 +02:00
|
|
|
};
|
2014-09-02 00:22:13 +02:00
|
|
|
|
2014-11-19 15:16:30 +01:00
|
|
|
//default pointer for HLE structures (BE pointer to BE data)
|
2015-01-19 17:30:35 +01:00
|
|
|
template<typename T, int lvl, typename AT> struct bptr : public bptrb<T, lvl, AT>
|
2014-09-02 00:22:13 +02:00
|
|
|
{
|
2015-01-19 22:30:33 +01:00
|
|
|
static const bptr make(AT addr)
|
2014-09-02 00:22:13 +02:00
|
|
|
{
|
2015-01-19 17:59:07 +01:00
|
|
|
auto res = bptrb<T, lvl, AT>::make(addr);
|
2015-01-19 22:30:33 +01:00
|
|
|
return static_cast<const bptr&>(res);
|
2015-01-19 17:30:35 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
vm::ps3::ptr<T, lvl, AT> to_le() const
|
|
|
|
|
{
|
2015-01-19 17:59:07 +01:00
|
|
|
return vm::ps3::ptr<T, lvl, AT>::make(this->addr());
|
2014-09-02 00:22:13 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
using bptrb<T, lvl, AT>::operator=;
|
|
|
|
|
};
|
2014-09-01 18:16:44 +02:00
|
|
|
}
|
2014-09-02 00:22:13 +02:00
|
|
|
|
2014-09-01 18:16:44 +02:00
|
|
|
namespace psv
|
|
|
|
|
{
|
2014-11-19 15:16:30 +01:00
|
|
|
//default pointer for HLE functions & structures (LE pointer to LE data)
|
2014-09-02 00:22:13 +02:00
|
|
|
template<typename T, int lvl = 1, typename AT = u32> struct ptr : public lptrl<T, lvl, AT>
|
|
|
|
|
{
|
2015-01-19 22:30:33 +01:00
|
|
|
static const ptr make(AT addr)
|
2014-09-02 00:22:13 +02:00
|
|
|
{
|
2015-01-19 17:59:07 +01:00
|
|
|
auto res = lptrl<T, lvl, AT>::make(addr);
|
2015-01-19 22:30:33 +01:00
|
|
|
return static_cast<const ptr&>(res);
|
2014-09-02 00:22:13 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
using lptrl<T, lvl, AT>::operator=;
|
|
|
|
|
};
|
2014-09-01 18:16:44 +02:00
|
|
|
}
|
2014-09-02 00:22:13 +02:00
|
|
|
|
2014-09-01 18:16:44 +02:00
|
|
|
//PS3 emulation is main now, so lets it be as default
|
|
|
|
|
using namespace ps3;
|
2015-01-13 18:14:07 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
namespace fmt
|
|
|
|
|
{
|
|
|
|
|
// external specializations for fmt::format function
|
2015-01-18 23:54:56 +01:00
|
|
|
|
|
|
|
|
template<typename T, int lvl, typename AT>
|
|
|
|
|
struct unveil<vm::ps3::ptr<T, lvl, AT>, false>
|
2015-01-13 18:14:07 +01:00
|
|
|
{
|
2015-01-18 23:54:56 +01:00
|
|
|
typedef typename unveil<AT>::result_type result_type;
|
|
|
|
|
|
|
|
|
|
__forceinline static result_type get_value(const vm::ps3::ptr<T, lvl, AT>& arg)
|
2015-01-13 18:14:07 +01:00
|
|
|
{
|
2015-01-18 23:54:56 +01:00
|
|
|
return unveil<AT>::get_value(arg.addr());
|
|
|
|
|
}
|
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
template<typename T, int lvl, typename AT>
|
|
|
|
|
struct unveil<vm::ps3::bptr<T, lvl, AT>, false>
|
|
|
|
|
{
|
|
|
|
|
typedef typename unveil<AT>::result_type result_type;
|
2015-01-13 18:14:07 +01:00
|
|
|
|
2015-01-18 23:54:56 +01:00
|
|
|
__forceinline static result_type get_value(const vm::ps3::bptr<T, lvl, AT>& arg)
|
2015-01-13 18:14:07 +01:00
|
|
|
{
|
2015-01-18 23:54:56 +01:00
|
|
|
return unveil<AT>::get_value(arg.addr());
|
|
|
|
|
}
|
|
|
|
|
};
|
2015-01-13 18:14:07 +01:00
|
|
|
|
2015-01-18 23:54:56 +01:00
|
|
|
template<typename T, int lvl, typename AT>
|
|
|
|
|
struct unveil<vm::psv::ptr<T, lvl, AT>, false>
|
|
|
|
|
{
|
|
|
|
|
typedef typename unveil<AT>::result_type result_type;
|
|
|
|
|
|
|
|
|
|
__forceinline static result_type get_value(const vm::psv::ptr<T, lvl, AT>& arg)
|
2015-01-13 18:14:07 +01:00
|
|
|
{
|
2015-01-18 23:54:56 +01:00
|
|
|
return unveil<AT>::get_value(arg.addr());
|
|
|
|
|
}
|
|
|
|
|
};
|
2015-01-13 18:14:07 +01:00
|
|
|
}
|
2015-01-19 15:16:31 +01:00
|
|
|
|
|
|
|
|
// external specializations for PPU GPR (SC_FUNC.h, CB_FUNC.h)
|
|
|
|
|
|
|
|
|
|
template<typename T, bool is_enum>
|
|
|
|
|
struct cast_ppu_gpr;
|
|
|
|
|
|
|
|
|
|
template<typename T, int lvl, typename AT>
|
|
|
|
|
struct cast_ppu_gpr<vm::ps3::ptr<T, lvl, AT>, false>
|
|
|
|
|
{
|
|
|
|
|
__forceinline static u64 to_gpr(const vm::ps3::ptr<T, lvl, AT>& value)
|
|
|
|
|
{
|
|
|
|
|
return value.addr();
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
__forceinline static vm::ps3::ptr<T, lvl, AT> from_gpr(const u64 reg)
|
|
|
|
|
{
|
2015-01-19 15:34:11 +01:00
|
|
|
return vm::ps3::ptr<T, lvl, AT>::make(cast_ppu_gpr<AT, std::is_enum<AT>::value>::from_gpr(reg));
|
2015-01-19 15:16:31 +01:00
|
|
|
}
|
|
|
|
|
};
|
2015-01-19 19:02:33 +01:00
|
|
|
|
|
|
|
|
// external specializations for ARMv7 GPR
|
|
|
|
|
|
|
|
|
|
template<typename T, bool is_enum>
|
|
|
|
|
struct cast_armv7_gpr;
|
|
|
|
|
|
|
|
|
|
template<typename T, int lvl, typename AT>
|
|
|
|
|
struct cast_armv7_gpr<vm::psv::ptr<T, lvl, AT>, false>
|
|
|
|
|
{
|
|
|
|
|
__forceinline static u32 to_gpr(const vm::psv::ptr<T, lvl, AT>& value)
|
|
|
|
|
{
|
|
|
|
|
return value.addr();
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
__forceinline static vm::psv::ptr<T, lvl, AT> from_gpr(const u32 reg)
|
|
|
|
|
{
|
|
|
|
|
return vm::psv::ptr<T, lvl, AT>::make(cast_armv7_gpr<AT, std::is_enum<AT>::value>::from_gpr(reg));
|
|
|
|
|
}
|
|
|
|
|
};
|