rpcsx/rpcs3/Emu/Memory/vm_ptr.h

497 lines
14 KiB
C
Raw Normal View History

#pragma once
2014-09-12 15:08:24 +02:00
class PPUThread;
namespace vm
{
template<typename T, int lvl = 1, typename AT = u32>
2014-09-01 18:16:44 +02:00
class _ptr_base
{
AT m_addr;
public:
typedef typename std::remove_cv<T>::type type;
2014-09-01 18:16:44 +02:00
_ptr_base operator++ (int)
{
AT result = m_addr;
m_addr += sizeof(AT);
2014-09-01 18:16:44 +02:00
return make(result);
}
2014-09-01 18:16:44 +02:00
_ptr_base& operator++ ()
{
m_addr += sizeof(AT);
return *this;
}
2014-09-01 18:16:44 +02:00
_ptr_base operator-- (int)
{
AT result = m_addr;
m_addr -= sizeof(AT);
2014-09-01 18:16:44 +02:00
return make(result);
}
2014-09-01 18:16:44 +02:00
_ptr_base& operator-- ()
{
m_addr -= sizeof(AT);
return *this;
}
_ptr_base& operator += (AT count)
{
m_addr += count * sizeof(AT);
return *this;
}
_ptr_base& operator -= (AT count)
{
m_addr -= count * sizeof(AT);
return *this;
}
_ptr_base operator + (typename remove_be_t<AT>::type count) const { return make(m_addr + count * sizeof(AT)); }
_ptr_base operator + (typename to_be_t<AT>::type count) const { return make(m_addr + count * sizeof(AT)); }
_ptr_base operator - (typename remove_be_t<AT>::type count) const { return make(m_addr - count * sizeof(AT)); }
_ptr_base operator - (typename to_be_t<AT>::type count) const { return make(m_addr - count * sizeof(AT)); }
__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; }
__forceinline _ptr_base<T, lvl - 1, std::conditional<is_be_t<T>::value, typename to_be_t<AT>::type, AT>>& operator *() const
{
return vm::get_ref<_ptr_base<T, lvl - 1, std::conditional<is_be_t<T>::value, typename to_be_t<AT>::type, AT>>>(m_addr);
}
__forceinline _ptr_base<T, lvl - 1, std::conditional<is_be_t<T>::value, typename to_be_t<AT>::type, AT>>& operator [](AT index) const
{
return vm::get_ref<_ptr_base<T, lvl - 1, std::conditional<is_be_t<T>::value, typename to_be_t<AT>::type, AT>>>(m_addr + sizeof(AT)* index);
}
2014-09-04 19:32:20 +02:00
//typedef typename invert_be_t<AT>::type AT2;
template<typename AT2>
operator const _ptr_base<T, lvl, AT2>() const
{
typename std::remove_const<AT2>::type addr = convert_le_be<AT2>(m_addr);
2014-09-04 19:32:20 +02:00
return (_ptr_base<T, lvl, AT2>&)addr;
}
AT addr() const
{
return m_addr;
}
2014-09-11 21:18:19 +02:00
void set(const AT value)
{
m_addr = value;
}
2014-09-01 18:16:44 +02:00
static _ptr_base make(AT addr)
{
2014-09-01 18:16:44 +02:00
return (_ptr_base&)addr;
}
2014-09-04 19:32:20 +02:00
_ptr_base& operator = (const _ptr_base& right) = default;
};
template<typename T, typename AT>
2014-09-01 18:16:44 +02:00
class _ptr_base<T, 1, AT>
{
AT m_addr;
public:
typedef typename std::remove_cv<T>::type type;
2014-09-04 19:32:20 +02:00
__forceinline T* const operator -> () const
{
2014-08-31 01:19:10 +02:00
return vm::get_ptr<T>(m_addr);
}
2014-09-01 18:16:44 +02:00
_ptr_base operator++ (int)
{
AT result = m_addr;
m_addr += sizeof(T);
2014-09-01 18:16:44 +02:00
return make(result);
}
2014-09-01 18:16:44 +02:00
_ptr_base& operator++ ()
{
m_addr += sizeof(T);
return *this;
}
2014-09-01 18:16:44 +02:00
_ptr_base operator-- (int)
{
AT result = m_addr;
m_addr -= sizeof(T);
2014-09-01 18:16:44 +02:00
return make(result);
}
2014-09-01 18:16:44 +02:00
_ptr_base& operator-- ()
{
m_addr -= sizeof(T);
return *this;
}
_ptr_base& operator += (AT count)
{
m_addr += count * sizeof(T);
return *this;
}
_ptr_base& operator -= (AT count)
{
m_addr -= count * sizeof(T);
return *this;
}
_ptr_base operator + (typename remove_be_t<AT>::type count) const { return make(m_addr + count * sizeof(T)); }
_ptr_base operator + (typename to_be_t<AT>::type count) const { return make(m_addr + count * sizeof(T)); }
_ptr_base operator - (typename remove_be_t<AT>::type count) const { return make(m_addr - count * sizeof(T)); }
_ptr_base operator - (typename to_be_t<AT>::type count) const { return make(m_addr - count * sizeof(T)); }
2014-09-04 19:32:20 +02:00
__forceinline T& operator *() const
{
return vm::get_ref<T>(m_addr);
}
__forceinline T& operator [](typename remove_be_t<AT>::type index) const
{
return vm::get_ref<T>(m_addr + sizeof(AT)* index);
}
__forceinline T& operator [](typename to_be_t<AT>::forced_type index) const
{
return vm::get_ref<T>(m_addr + sizeof(AT)* index);
}
__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-09-01 18:16:44 +02:00
operator _ref_base<T, AT>()
{
2014-09-01 18:16:44 +02:00
return _ref_base<T, AT>::make(m_addr);
}
2014-09-01 18:16:44 +02:00
operator const _ref_base<T, AT>() const
{
2014-09-01 18:16:44 +02:00
return _ref_base<T, AT>::make(m_addr);
}
*/
AT addr() const
{
return m_addr;
}
template<typename T>
void set(T&& value)
2014-09-11 21:18:19 +02:00
{
m_addr = convert_le_be<AT>(value);
2014-09-11 21:18:19 +02:00
}
/*
operator T*() const
{
return get_ptr();
}
*/
2014-09-04 19:32:20 +02:00
//typedef typename invert_be_t<AT>::type AT2;
template<typename AT2>
operator const _ptr_base<T, 1, AT2>() const
{
typename std::remove_const<AT2>::type addr = convert_le_be<AT2>(m_addr);
2014-09-04 19:32:20 +02:00
return (_ptr_base<T, 1, AT2>&)addr;
}
T* get_ptr() const
{
return vm::get_ptr<T>(m_addr);
}
2014-09-01 18:16:44 +02:00
static _ptr_base make(AT addr)
{
2014-09-01 18:16:44 +02:00
return (_ptr_base&)addr;
}
2014-09-04 19:32:20 +02:00
_ptr_base& operator = (const _ptr_base& right) = default;
};
template<typename AT>
2014-09-01 18:16:44 +02:00
class _ptr_base<void, 1, AT>
{
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;
}
void* get_ptr() const
{
return vm::get_ptr<void>(m_addr);
}
explicit operator void*() const
{
return get_ptr();
}
__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
//typedef typename invert_be_t<AT>::type AT2;
template<typename AT2>
operator const _ptr_base<void, 1, AT2>() const
{
typename std::remove_const<AT2>::type addr = convert_le_be<AT2>(m_addr);
2014-09-04 19:32:20 +02:00
return (_ptr_base<void, 1, AT2>&)addr;
}
2014-09-08 02:54:17 +02:00
template<typename AT2>
operator const _ptr_base<const void, 1, AT2>() const
{
typename std::remove_const<AT2>::type addr = convert_le_be<AT2>(m_addr);
2014-09-08 02:54:17 +02:00
return (_ptr_base<const void, 1, AT2>&)addr;
}
2014-09-01 18:16:44 +02:00
static _ptr_base make(AT addr)
{
2014-09-01 18:16:44 +02:00
return (_ptr_base&)addr;
}
2014-09-04 19:32:20 +02:00
_ptr_base& operator = (const _ptr_base& right) = default;
};
template<typename AT>
2014-09-01 18:16:44 +02:00
class _ptr_base<const void, 1, AT>
{
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;
}
const void* get_ptr() const
{
return vm::get_ptr<const void>(m_addr);
}
explicit operator const void*() const
{
return get_ptr();
}
__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
//typedef typename invert_be_t<AT>::type AT2;
template<typename AT2>
operator const _ptr_base<const void, 1, AT2>() const
{
typename std::remove_const<AT2>::type addr = convert_le_be<AT2>(m_addr);
2014-09-04 19:32:20 +02:00
return (_ptr_base<const void, 1, AT2>&)addr;
}
2014-09-01 18:16:44 +02:00
static _ptr_base make(AT addr)
{
2014-09-01 18:16:44 +02:00
return (_ptr_base&)addr;
}
2014-09-04 19:32:20 +02:00
_ptr_base& operator = (const _ptr_base& right) = default;
};
template<typename AT, typename RT, typename ...T>
2014-09-01 18:16:44 +02:00
class _ptr_base<RT(*)(T...), 1, AT>
{
AT m_addr;
public:
typedef RT(*type)(T...);
2014-09-12 15:08:24 +02:00
RT call(PPUThread& CPU, T... args) const; // call using specified PPU thread context, defined in Callback.h (CB_FUNC.h)
RT operator()(T... args) const; // call using current PPU thread context, defined in Callback.h (CB_FUNC.h)
2014-09-11 21:18:19 +02:00
AT addr() const
{
2014-09-11 21:18:19 +02:00
return m_addr;
}
2014-09-11 21:18:19 +02:00
void set(const AT value)
{
2014-09-11 21:18:19 +02:00
m_addr = value;
}
__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
//typedef typename invert_be_t<AT>::type AT2;
template<typename AT2>
operator const _ptr_base<RT(*)(T...), 1, AT2>() const
{
typename std::remove_const<AT2>::type addr = convert_le_be<AT2>(m_addr);
2014-09-04 19:32:20 +02:00
return (_ptr_base<RT(*)(T...), 1, AT2>&)addr;
}
2014-09-01 18:16:44 +02:00
static _ptr_base make(AT addr)
{
2014-09-01 18:16:44 +02:00
return (_ptr_base&)addr;
}
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
{
typename std::remove_const<AT>::type 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-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>
{
static bptrl make(AT addr)
2014-09-02 00:22:13 +02:00
{
return (bptrl&)addr;
}
using _ptr_base<T, lvl, typename to_be_t<AT>::type>::operator=;
2014-09-04 19:32:20 +02:00
//using _ptr_base<T, lvl, typename to_be_t<AT>::type>::operator const _ptr_base<T, lvl, AT>;
2014-09-02 00:22:13 +02:00
};
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>
{
static bptrb make(AT addr)
2014-09-02 00:22:13 +02:00
{
return (bptrb&)addr;
}
using _ptr_base<typename to_be_t<T>::type, lvl, typename to_be_t<AT>::type>::operator=;
2014-09-04 19:32:20 +02:00
//using _ptr_base<typename to_be_t<T>::type, lvl, typename to_be_t<AT>::type>::operator const _ptr_base<typename to_be_t<T>::type, lvl, AT>;
2014-09-02 00:22:13 +02:00
};
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>
{
static lptrb make(AT addr)
{
return (lptrb&)addr;
}
using _ptr_base<typename to_be_t<T>::type, lvl, AT>::operator=;
2014-09-04 19:32:20 +02:00
//using _ptr_base<typename to_be_t<T>::type, lvl, AT>::operator const _ptr_base<typename to_be_t<T>::type, lvl, typename to_be_t<AT>::type>;
2014-09-02 00:22:13 +02:00
};
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>
{
static lptrl make(AT addr)
{
return (lptrl&)addr;
}
using _ptr_base<T, lvl, AT>::operator=;
2014-09-04 19:32:20 +02:00
//using _ptr_base<T, lvl, AT>::operator const _ptr_base<T, lvl, typename to_be_t<AT>::type>;
2014-09-02 00:22:13 +02:00
};
2014-09-01 18:16:44 +02:00
namespace ps3
{
//default pointer for HLE functions (LE pointer to BE data)
2014-09-02 00:22:13 +02:00
template<typename T, int lvl = 1, typename AT = u32> struct ptr : public lptrb<T, lvl, AT>
2014-09-01 18:16:44 +02:00
{
static ptr make(AT addr)
{
return (ptr&)addr;
}
2014-09-02 00:22:13 +02:00
using lptrb<T, lvl, AT>::operator=;
2014-09-04 19:32:20 +02:00
//using lptrb<T, lvl, AT>::operator const _ptr_base<typename to_be_t<T>::type, lvl, AT>;
2014-09-01 18:16:44 +02:00
};
2014-09-02 00:22:13 +02:00
//default pointer for HLE structures (BE pointer to BE data)
2014-09-02 00:22:13 +02:00
template<typename T, int lvl = 1, typename AT = u32> struct bptr : public bptrb<T, lvl, AT>
{
static bptr make(AT addr)
2014-09-02 00:22:13 +02:00
{
return (bptr&)addr;
}
using bptrb<T, lvl, AT>::operator=;
2014-09-04 19:32:20 +02:00
//using bptrb<T, lvl, AT>::operator const _ptr_base<typename to_be_t<T>::type, lvl, AT>;
2014-09-02 00:22:13 +02:00
};
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
{
//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>
{
static ptr make(AT addr)
{
return (ptr&)addr;
}
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;
}