2013-04-17 23:09:55 -04:00
|
|
|
// Copyright 2013 Dolphin Emulator Project
|
|
|
|
// Licensed under GPLv2
|
|
|
|
// Refer to the license.txt file included.
|
2008-12-08 05:25:12 +00:00
|
|
|
|
2008-12-26 13:03:50 +00:00
|
|
|
#include "Common.h"
|
|
|
|
#include "VideoCommon.h"
|
2008-12-08 05:25:12 +00:00
|
|
|
#include "VertexLoader.h"
|
|
|
|
#include "VertexLoader_Normal.h"
|
2010-10-03 00:41:06 +00:00
|
|
|
#include "VertexManagerBase.h"
|
2010-04-09 15:13:42 +00:00
|
|
|
#include "CPUDetect.h"
|
2011-01-23 15:29:57 +00:00
|
|
|
#include <cmath>
|
2013-02-20 22:22:41 -06:00
|
|
|
#include <limits>
|
2008-12-08 05:25:12 +00:00
|
|
|
|
2011-01-23 15:29:57 +00:00
|
|
|
#if _M_SSE >= 0x401
|
|
|
|
#include <smmintrin.h>
|
|
|
|
#include <emmintrin.h>
|
|
|
|
#elif _M_SSE >= 0x301 && !(defined __GNUC__ && !defined __SSSE3__)
|
2010-04-09 03:02:12 +00:00
|
|
|
#include <tmmintrin.h>
|
|
|
|
#endif
|
|
|
|
|
2013-02-21 11:45:29 +01:00
|
|
|
// warning: mapping buffer should be disabled to use this
|
2011-01-23 15:29:57 +00:00
|
|
|
#define LOG_NORM() // PRIM_LOG("norm: %f %f %f, ", ((float*)VertexManager::s_pCurBufferPointer)[-3], ((float*)VertexManager::s_pCurBufferPointer)[-2], ((float*)VertexManager::s_pCurBufferPointer)[-1]);
|
2008-12-08 05:25:12 +00:00
|
|
|
|
|
|
|
VertexLoader_Normal::Set VertexLoader_Normal::m_Table[NUM_NRM_TYPE][NUM_NRM_INDICES][NUM_NRM_ELEMENTS][NUM_NRM_FORMAT];
|
|
|
|
|
2013-02-20 22:22:41 -06:00
|
|
|
namespace
|
2008-12-08 05:25:12 +00:00
|
|
|
{
|
|
|
|
|
2013-02-20 22:22:41 -06:00
|
|
|
template <typename T>
|
2013-02-21 13:25:35 -06:00
|
|
|
__forceinline float FracAdjust(T val)
|
2008-12-08 05:25:12 +00:00
|
|
|
{
|
2013-02-20 22:22:41 -06:00
|
|
|
//auto const S8FRAC = 1.f / (1u << 6);
|
|
|
|
//auto const U8FRAC = 1.f / (1u << 7);
|
|
|
|
//auto const S16FRAC = 1.f / (1u << 14);
|
|
|
|
//auto const U16FRAC = 1.f / (1u << 15);
|
2013-03-19 21:51:12 -04:00
|
|
|
|
2013-02-21 00:49:47 -06:00
|
|
|
// TODO: is this right?
|
2013-02-20 22:22:41 -06:00
|
|
|
return val / float(1u << (sizeof(T) * 8 - std::numeric_limits<T>::is_signed - 1));
|
2008-12-08 05:25:12 +00:00
|
|
|
}
|
|
|
|
|
2013-02-20 22:22:41 -06:00
|
|
|
template <>
|
2013-02-21 13:25:35 -06:00
|
|
|
__forceinline float FracAdjust(float val)
|
2013-04-24 09:21:54 -04:00
|
|
|
{
|
|
|
|
return val;
|
|
|
|
}
|
2009-09-13 21:18:04 +00:00
|
|
|
|
2013-02-20 22:22:41 -06:00
|
|
|
template <typename T, int N>
|
2013-02-21 13:25:35 -06:00
|
|
|
__forceinline void ReadIndirect(const T* data)
|
2008-12-08 05:25:12 +00:00
|
|
|
{
|
2013-02-20 22:22:41 -06:00
|
|
|
static_assert(3 == N || 9 == N, "N is only sane as 3 or 9!");
|
2014-01-21 23:44:51 +01:00
|
|
|
DataWriter dst;
|
2008-12-08 05:25:12 +00:00
|
|
|
|
2013-02-20 22:22:41 -06:00
|
|
|
for (int i = 0; i != N; ++i)
|
2008-12-08 05:25:12 +00:00
|
|
|
{
|
2014-01-21 23:44:51 +01:00
|
|
|
dst.Write(FracAdjust(Common::FromBigEndian(data[i])));
|
2013-02-20 22:22:41 -06:00
|
|
|
}
|
2013-03-19 21:51:12 -04:00
|
|
|
|
2013-02-21 13:45:48 +01:00
|
|
|
LOG_NORM();
|
2009-09-13 21:18:04 +00:00
|
|
|
}
|
|
|
|
|
2013-02-20 22:22:41 -06:00
|
|
|
template <typename T, int N>
|
2013-02-21 13:25:35 -06:00
|
|
|
struct Normal_Direct
|
2008-12-08 05:25:12 +00:00
|
|
|
{
|
2013-02-21 13:25:35 -06:00
|
|
|
static void LOADERDECL function()
|
|
|
|
{
|
|
|
|
auto const source = reinterpret_cast<const T*>(DataGetPosition());
|
|
|
|
ReadIndirect<T, N * 3>(source);
|
|
|
|
DataSkip<N * 3 * sizeof(T)>();
|
|
|
|
}
|
2013-03-19 21:51:12 -04:00
|
|
|
|
2013-02-21 13:25:35 -06:00
|
|
|
static const int size = sizeof(T) * N * 3;
|
|
|
|
};
|
2008-12-08 05:25:12 +00:00
|
|
|
|
2013-02-20 22:22:41 -06:00
|
|
|
template <typename I, typename T, int N, int Offset>
|
2013-02-21 13:25:35 -06:00
|
|
|
__forceinline void Normal_Index_Offset()
|
2008-12-08 05:25:12 +00:00
|
|
|
{
|
2013-02-20 22:22:41 -06:00
|
|
|
static_assert(!std::numeric_limits<I>::is_signed, "Only unsigned I is sane!");
|
2013-10-29 01:23:17 -04:00
|
|
|
|
2013-02-20 22:22:41 -06:00
|
|
|
auto const index = DataRead<I>();
|
|
|
|
auto const data = reinterpret_cast<const T*>(cached_arraybases[ARRAY_NORMAL]
|
|
|
|
+ (index * arraystrides[ARRAY_NORMAL]) + sizeof(T) * 3 * Offset);
|
|
|
|
ReadIndirect<T, N * 3>(data);
|
2008-12-08 05:25:12 +00:00
|
|
|
}
|
|
|
|
|
2013-02-20 22:22:41 -06:00
|
|
|
template <typename I, typename T, int N>
|
2013-02-21 13:25:35 -06:00
|
|
|
struct Normal_Index
|
2009-09-13 21:18:04 +00:00
|
|
|
{
|
2013-02-21 13:25:35 -06:00
|
|
|
static void LOADERDECL function()
|
|
|
|
{
|
|
|
|
Normal_Index_Offset<I, T, N, 0>();
|
|
|
|
}
|
2013-03-19 21:51:12 -04:00
|
|
|
|
2013-02-21 13:25:35 -06:00
|
|
|
static const int size = sizeof(I);
|
|
|
|
};
|
2009-09-13 21:18:04 +00:00
|
|
|
|
2013-02-20 22:22:41 -06:00
|
|
|
template <typename I, typename T>
|
2013-02-21 13:25:35 -06:00
|
|
|
struct Normal_Index_Indices3
|
2008-12-08 05:25:12 +00:00
|
|
|
{
|
2013-02-21 13:25:35 -06:00
|
|
|
static void LOADERDECL function()
|
|
|
|
{
|
|
|
|
Normal_Index_Offset<I, T, 1, 0>();
|
|
|
|
Normal_Index_Offset<I, T, 1, 1>();
|
|
|
|
Normal_Index_Offset<I, T, 1, 2>();
|
|
|
|
}
|
2013-03-19 21:51:12 -04:00
|
|
|
|
2013-02-21 13:25:35 -06:00
|
|
|
static const int size = sizeof(I) * 3;
|
|
|
|
};
|
2008-12-08 05:25:12 +00:00
|
|
|
|
|
|
|
}
|
|
|
|
|
2013-02-20 22:22:41 -06:00
|
|
|
void VertexLoader_Normal::Init(void)
|
2008-12-08 05:25:12 +00:00
|
|
|
{
|
2013-02-21 13:25:35 -06:00
|
|
|
m_Table[NRM_DIRECT] [NRM_INDICES1][NRM_NBT] [FORMAT_UBYTE] = Normal_Direct<u8, 1>();
|
|
|
|
m_Table[NRM_DIRECT] [NRM_INDICES1][NRM_NBT] [FORMAT_BYTE] = Normal_Direct<s8, 1>();
|
|
|
|
m_Table[NRM_DIRECT] [NRM_INDICES1][NRM_NBT] [FORMAT_USHORT] = Normal_Direct<u16, 1>();
|
|
|
|
m_Table[NRM_DIRECT] [NRM_INDICES1][NRM_NBT] [FORMAT_SHORT] = Normal_Direct<s16, 1>();
|
|
|
|
m_Table[NRM_DIRECT] [NRM_INDICES1][NRM_NBT] [FORMAT_FLOAT] = Normal_Direct<float, 1>();
|
|
|
|
m_Table[NRM_DIRECT] [NRM_INDICES1][NRM_NBT3][FORMAT_UBYTE] = Normal_Direct<u8, 3>();
|
|
|
|
m_Table[NRM_DIRECT] [NRM_INDICES1][NRM_NBT3][FORMAT_BYTE] = Normal_Direct<s8, 3>();
|
|
|
|
m_Table[NRM_DIRECT] [NRM_INDICES1][NRM_NBT3][FORMAT_USHORT] = Normal_Direct<u16, 3>();
|
|
|
|
m_Table[NRM_DIRECT] [NRM_INDICES1][NRM_NBT3][FORMAT_SHORT] = Normal_Direct<s16, 3>();
|
|
|
|
m_Table[NRM_DIRECT] [NRM_INDICES1][NRM_NBT3][FORMAT_FLOAT] = Normal_Direct<float, 3>();
|
|
|
|
|
|
|
|
// Same as above
|
|
|
|
m_Table[NRM_DIRECT] [NRM_INDICES3][NRM_NBT] [FORMAT_UBYTE] = Normal_Direct<u8, 1>();
|
|
|
|
m_Table[NRM_DIRECT] [NRM_INDICES3][NRM_NBT] [FORMAT_BYTE] = Normal_Direct<s8, 1>();
|
|
|
|
m_Table[NRM_DIRECT] [NRM_INDICES3][NRM_NBT] [FORMAT_USHORT] = Normal_Direct<u16, 1>();
|
|
|
|
m_Table[NRM_DIRECT] [NRM_INDICES3][NRM_NBT] [FORMAT_SHORT] = Normal_Direct<s16, 1>();
|
|
|
|
m_Table[NRM_DIRECT] [NRM_INDICES3][NRM_NBT] [FORMAT_FLOAT] = Normal_Direct<float, 1>();
|
|
|
|
m_Table[NRM_DIRECT] [NRM_INDICES3][NRM_NBT3][FORMAT_UBYTE] = Normal_Direct<u8, 3>();
|
|
|
|
m_Table[NRM_DIRECT] [NRM_INDICES3][NRM_NBT3][FORMAT_BYTE] = Normal_Direct<s8, 3>();
|
|
|
|
m_Table[NRM_DIRECT] [NRM_INDICES3][NRM_NBT3][FORMAT_USHORT] = Normal_Direct<u16, 3>();
|
|
|
|
m_Table[NRM_DIRECT] [NRM_INDICES3][NRM_NBT3][FORMAT_SHORT] = Normal_Direct<s16, 3>();
|
|
|
|
m_Table[NRM_DIRECT] [NRM_INDICES3][NRM_NBT3][FORMAT_FLOAT] = Normal_Direct<float, 3>();
|
|
|
|
|
|
|
|
m_Table[NRM_INDEX8] [NRM_INDICES1][NRM_NBT] [FORMAT_UBYTE] = Normal_Index<u8, u8, 1>();
|
|
|
|
m_Table[NRM_INDEX8] [NRM_INDICES1][NRM_NBT] [FORMAT_BYTE] = Normal_Index<u8, s8, 1>();
|
|
|
|
m_Table[NRM_INDEX8] [NRM_INDICES1][NRM_NBT] [FORMAT_USHORT] = Normal_Index<u8, u16, 1>();
|
|
|
|
m_Table[NRM_INDEX8] [NRM_INDICES1][NRM_NBT] [FORMAT_SHORT] = Normal_Index<u8, s16, 1>();
|
|
|
|
m_Table[NRM_INDEX8] [NRM_INDICES1][NRM_NBT] [FORMAT_FLOAT] = Normal_Index<u8, float, 1>();
|
|
|
|
m_Table[NRM_INDEX8] [NRM_INDICES1][NRM_NBT3][FORMAT_UBYTE] = Normal_Index<u8, u8, 3>();
|
|
|
|
m_Table[NRM_INDEX8] [NRM_INDICES1][NRM_NBT3][FORMAT_BYTE] = Normal_Index<u8, s8, 3>();
|
|
|
|
m_Table[NRM_INDEX8] [NRM_INDICES1][NRM_NBT3][FORMAT_USHORT] = Normal_Index<u8, u16, 3>();
|
|
|
|
m_Table[NRM_INDEX8] [NRM_INDICES1][NRM_NBT3][FORMAT_SHORT] = Normal_Index<u8, s16, 3>();
|
|
|
|
m_Table[NRM_INDEX8] [NRM_INDICES1][NRM_NBT3][FORMAT_FLOAT] = Normal_Index<u8, float, 3>();
|
|
|
|
|
|
|
|
// Same as above for NRM_NBT
|
|
|
|
m_Table[NRM_INDEX8] [NRM_INDICES3][NRM_NBT] [FORMAT_UBYTE] = Normal_Index<u8, u8, 1>();
|
|
|
|
m_Table[NRM_INDEX8] [NRM_INDICES3][NRM_NBT] [FORMAT_BYTE] = Normal_Index<u8, s8, 1>();
|
|
|
|
m_Table[NRM_INDEX8] [NRM_INDICES3][NRM_NBT] [FORMAT_USHORT] = Normal_Index<u8, u16, 1>();
|
|
|
|
m_Table[NRM_INDEX8] [NRM_INDICES3][NRM_NBT] [FORMAT_SHORT] = Normal_Index<u8, s16, 1>();
|
|
|
|
m_Table[NRM_INDEX8] [NRM_INDICES3][NRM_NBT] [FORMAT_FLOAT] = Normal_Index<u8, float, 1>();
|
|
|
|
m_Table[NRM_INDEX8] [NRM_INDICES3][NRM_NBT3][FORMAT_UBYTE] = Normal_Index_Indices3<u8, u8>();
|
|
|
|
m_Table[NRM_INDEX8] [NRM_INDICES3][NRM_NBT3][FORMAT_BYTE] = Normal_Index_Indices3<u8, s8>();
|
|
|
|
m_Table[NRM_INDEX8] [NRM_INDICES3][NRM_NBT3][FORMAT_USHORT] = Normal_Index_Indices3<u8, u16>();
|
|
|
|
m_Table[NRM_INDEX8] [NRM_INDICES3][NRM_NBT3][FORMAT_SHORT] = Normal_Index_Indices3<u8, s16>();
|
|
|
|
m_Table[NRM_INDEX8] [NRM_INDICES3][NRM_NBT3][FORMAT_FLOAT] = Normal_Index_Indices3<u8, float>();
|
|
|
|
|
|
|
|
m_Table[NRM_INDEX16][NRM_INDICES1][NRM_NBT] [FORMAT_UBYTE] = Normal_Index<u16, u8, 1>();
|
|
|
|
m_Table[NRM_INDEX16][NRM_INDICES1][NRM_NBT] [FORMAT_BYTE] = Normal_Index<u16, s8, 1>();
|
|
|
|
m_Table[NRM_INDEX16][NRM_INDICES1][NRM_NBT] [FORMAT_USHORT] = Normal_Index<u16, u16, 1>();
|
|
|
|
m_Table[NRM_INDEX16][NRM_INDICES1][NRM_NBT] [FORMAT_SHORT] = Normal_Index<u16, s16, 1>();
|
|
|
|
m_Table[NRM_INDEX16][NRM_INDICES1][NRM_NBT] [FORMAT_FLOAT] = Normal_Index<u16, float, 1>();
|
|
|
|
m_Table[NRM_INDEX16][NRM_INDICES1][NRM_NBT3][FORMAT_UBYTE] = Normal_Index<u16, u8, 3>();
|
|
|
|
m_Table[NRM_INDEX16][NRM_INDICES1][NRM_NBT3][FORMAT_BYTE] = Normal_Index<u16, s8, 3>();
|
|
|
|
m_Table[NRM_INDEX16][NRM_INDICES1][NRM_NBT3][FORMAT_USHORT] = Normal_Index<u16, u16, 3>();
|
|
|
|
m_Table[NRM_INDEX16][NRM_INDICES1][NRM_NBT3][FORMAT_SHORT] = Normal_Index<u16, s16, 3>();
|
|
|
|
m_Table[NRM_INDEX16][NRM_INDICES1][NRM_NBT3][FORMAT_FLOAT] = Normal_Index<u16, float, 3>();
|
|
|
|
|
|
|
|
// Same as above for NRM_NBT
|
|
|
|
m_Table[NRM_INDEX16][NRM_INDICES3][NRM_NBT] [FORMAT_UBYTE] = Normal_Index<u16, u8, 1>();
|
|
|
|
m_Table[NRM_INDEX16][NRM_INDICES3][NRM_NBT] [FORMAT_BYTE] = Normal_Index<u16, s8, 1>();
|
|
|
|
m_Table[NRM_INDEX16][NRM_INDICES3][NRM_NBT] [FORMAT_USHORT] = Normal_Index<u16, u16, 1>();
|
|
|
|
m_Table[NRM_INDEX16][NRM_INDICES3][NRM_NBT] [FORMAT_SHORT] = Normal_Index<u16, s16, 1>();
|
|
|
|
m_Table[NRM_INDEX16][NRM_INDICES3][NRM_NBT] [FORMAT_FLOAT] = Normal_Index<u16, float, 1>();
|
|
|
|
m_Table[NRM_INDEX16][NRM_INDICES3][NRM_NBT3][FORMAT_UBYTE] = Normal_Index_Indices3<u16, u8>();
|
|
|
|
m_Table[NRM_INDEX16][NRM_INDICES3][NRM_NBT3][FORMAT_BYTE] = Normal_Index_Indices3<u16, s8>();
|
|
|
|
m_Table[NRM_INDEX16][NRM_INDICES3][NRM_NBT3][FORMAT_USHORT] = Normal_Index_Indices3<u16, u16>();
|
|
|
|
m_Table[NRM_INDEX16][NRM_INDICES3][NRM_NBT3][FORMAT_SHORT] = Normal_Index_Indices3<u16, s16>();
|
|
|
|
m_Table[NRM_INDEX16][NRM_INDICES3][NRM_NBT3][FORMAT_FLOAT] = Normal_Index_Indices3<u16, float>();
|
2008-12-08 05:25:12 +00:00
|
|
|
}
|
|
|
|
|
2013-02-20 22:22:41 -06:00
|
|
|
unsigned int VertexLoader_Normal::GetSize(unsigned int _type,
|
|
|
|
unsigned int _format, unsigned int _elements, unsigned int _index3)
|
2008-12-08 05:25:12 +00:00
|
|
|
{
|
2013-02-20 22:22:41 -06:00
|
|
|
return m_Table[_type][_index3][_elements][_format].gc_size;
|
2008-12-08 05:25:12 +00:00
|
|
|
}
|
|
|
|
|
2013-02-20 22:22:41 -06:00
|
|
|
TPipelineFunction VertexLoader_Normal::GetFunction(unsigned int _type,
|
|
|
|
unsigned int _format, unsigned int _elements, unsigned int _index3)
|
2008-12-08 05:25:12 +00:00
|
|
|
{
|
2013-02-20 22:22:41 -06:00
|
|
|
TPipelineFunction pFunc = m_Table[_type][_index3][_elements][_format].function;
|
2013-04-24 09:21:54 -04:00
|
|
|
return pFunc;
|
2008-12-08 05:25:12 +00:00
|
|
|
}
|