// Copyright 2009 Dolphin Emulator Project // Licensed under GPLv2+ // Refer to the license.txt file included. #include "VideoBackends/Software/SWVertexLoader.h" #include <limits> #include "Common/Assert.h" #include "Common/CommonTypes.h" #include "Common/Logging/Log.h" #include "VideoBackends/Software/DebugUtil.h" #include "VideoBackends/Software/NativeVertexFormat.h" #include "VideoBackends/Software/Rasterizer.h" #include "VideoBackends/Software/Tev.h" #include "VideoBackends/Software/TransformUnit.h" #include "VideoCommon/DataReader.h" #include "VideoCommon/IndexGenerator.h" #include "VideoCommon/OpcodeDecoding.h" #include "VideoCommon/PixelShaderManager.h" #include "VideoCommon/Statistics.h" #include "VideoCommon/VertexLoaderBase.h" #include "VideoCommon/VertexLoaderManager.h" #include "VideoCommon/VideoConfig.h" #include "VideoCommon/XFMemory.h" class NullNativeVertexFormat : public NativeVertexFormat { public: NullNativeVertexFormat(const PortableVertexDeclaration& _vtx_decl) { vtx_decl = _vtx_decl; } void SetupVertexPointers() override {} }; std::unique_ptr<NativeVertexFormat> SWVertexLoader::CreateNativeVertexFormat(const PortableVertexDeclaration& vtx_decl) { return std::make_unique<NullNativeVertexFormat>(vtx_decl); } SWVertexLoader::SWVertexLoader() : LocalVBuffer(MAXVBUFFERSIZE), LocalIBuffer(MAXIBUFFERSIZE) { } SWVertexLoader::~SWVertexLoader() { } void SWVertexLoader::ResetBuffer(u32 stride) { m_cur_buffer_pointer = m_base_buffer_pointer = LocalVBuffer.data(); m_end_buffer_pointer = m_cur_buffer_pointer + LocalVBuffer.size(); IndexGenerator::Start(GetIndexBuffer()); } void SWVertexLoader::vFlush() { DebugUtil::OnObjectBegin(); u8 primitiveType = 0; switch (m_current_primitive_type) { case PRIMITIVE_POINTS: primitiveType = OpcodeDecoder::GX_DRAW_POINTS; break; case PRIMITIVE_LINES: primitiveType = OpcodeDecoder::GX_DRAW_LINES; break; case PRIMITIVE_TRIANGLES: primitiveType = g_ActiveConfig.backend_info.bSupportsPrimitiveRestart ? OpcodeDecoder::GX_DRAW_TRIANGLE_STRIP : OpcodeDecoder::GX_DRAW_TRIANGLES; break; } m_SetupUnit.Init(primitiveType); // set all states with are stored within video sw for (int i = 0; i < 4; i++) { Rasterizer::SetTevReg(i, Tev::RED_C, PixelShaderManager::constants.kcolors[i][0]); Rasterizer::SetTevReg(i, Tev::GRN_C, PixelShaderManager::constants.kcolors[i][1]); Rasterizer::SetTevReg(i, Tev::BLU_C, PixelShaderManager::constants.kcolors[i][2]); Rasterizer::SetTevReg(i, Tev::ALP_C, PixelShaderManager::constants.kcolors[i][3]); } for (u32 i = 0; i < IndexGenerator::GetIndexLen(); i++) { u16 index = LocalIBuffer[i]; if (index == 0xffff) { // primitive restart m_SetupUnit.Init(primitiveType); continue; } memset(&m_Vertex, 0, sizeof(m_Vertex)); // Super Mario Sunshine requires those to be zero for those debug boxes. memset(&m_Vertex.color, 0, sizeof(m_Vertex.color)); // parse the videocommon format to our own struct format (m_Vertex) SetFormat(g_main_cp_state.last_id, primitiveType); ParseVertex(VertexLoaderManager::GetCurrentVertexFormat()->GetVertexDeclaration(), index); // transform this vertex so that it can be used for rasterization (outVertex) OutputVertexData* outVertex = m_SetupUnit.GetVertex(); TransformUnit::TransformPosition(&m_Vertex, outVertex); memset(&outVertex->normal, 0, sizeof(outVertex->normal)); if (VertexLoaderManager::g_current_components & VB_HAS_NRM0) { TransformUnit::TransformNormal( &m_Vertex, (VertexLoaderManager::g_current_components & VB_HAS_NRM2) != 0, outVertex); } TransformUnit::TransformColor(&m_Vertex, outVertex); TransformUnit::TransformTexCoord(&m_Vertex, outVertex, m_TexGenSpecialCase); // assemble and rasterize the primitive m_SetupUnit.SetupVertex(); INCSTAT(stats.thisFrame.numVerticesLoaded) } DebugUtil::OnObjectEnd(); } void SWVertexLoader::SetFormat(u8 attributeIndex, u8 primitiveType) { // matrix index from xf regs or cp memory? if (xfmem.MatrixIndexA.PosNormalMtxIdx != g_main_cp_state.matrix_index_a.PosNormalMtxIdx || xfmem.MatrixIndexA.Tex0MtxIdx != g_main_cp_state.matrix_index_a.Tex0MtxIdx || xfmem.MatrixIndexA.Tex1MtxIdx != g_main_cp_state.matrix_index_a.Tex1MtxIdx || xfmem.MatrixIndexA.Tex2MtxIdx != g_main_cp_state.matrix_index_a.Tex2MtxIdx || xfmem.MatrixIndexA.Tex3MtxIdx != g_main_cp_state.matrix_index_a.Tex3MtxIdx || xfmem.MatrixIndexB.Tex4MtxIdx != g_main_cp_state.matrix_index_b.Tex4MtxIdx || xfmem.MatrixIndexB.Tex5MtxIdx != g_main_cp_state.matrix_index_b.Tex5MtxIdx || xfmem.MatrixIndexB.Tex6MtxIdx != g_main_cp_state.matrix_index_b.Tex6MtxIdx || xfmem.MatrixIndexB.Tex7MtxIdx != g_main_cp_state.matrix_index_b.Tex7MtxIdx) { ERROR_LOG(VIDEO, "Matrix indices don't match"); } m_Vertex.posMtx = xfmem.MatrixIndexA.PosNormalMtxIdx; m_Vertex.texMtx[0] = xfmem.MatrixIndexA.Tex0MtxIdx; m_Vertex.texMtx[1] = xfmem.MatrixIndexA.Tex1MtxIdx; m_Vertex.texMtx[2] = xfmem.MatrixIndexA.Tex2MtxIdx; m_Vertex.texMtx[3] = xfmem.MatrixIndexA.Tex3MtxIdx; m_Vertex.texMtx[4] = xfmem.MatrixIndexB.Tex4MtxIdx; m_Vertex.texMtx[5] = xfmem.MatrixIndexB.Tex5MtxIdx; m_Vertex.texMtx[6] = xfmem.MatrixIndexB.Tex6MtxIdx; m_Vertex.texMtx[7] = xfmem.MatrixIndexB.Tex7MtxIdx; // special case if only pos and tex coord 0 and tex coord input is AB11 // http://libogc.devkitpro.org/gx_8h.html#a55a426a3ff796db584302bddd829f002 m_TexGenSpecialCase = VertexLoaderManager::g_current_components == VB_HAS_UV0 && xfmem.texMtxInfo[0].projection == XF_TEXPROJ_ST; } template <typename T, typename I> static T ReadNormalized(I value) { T casted = (T)value; if (!std::numeric_limits<T>::is_integer && std::numeric_limits<I>::is_integer) { // normalize if non-float is converted to a float casted *= (T)(1.0 / std::numeric_limits<I>::max()); } return casted; } template <typename T, bool swap = false> static void ReadVertexAttribute(T* dst, DataReader src, const AttributeFormat& format, int base_component, int components, bool reverse) { if (format.enable) { src.Skip(format.offset); src.Skip(base_component * (1 << (format.type >> 1))); int i; for (i = 0; i < std::min(format.components - base_component, components); i++) { int i_dst = reverse ? components - i - 1 : i; switch (format.type) { case VAR_UNSIGNED_BYTE: dst[i_dst] = ReadNormalized<T, u8>(src.Read<u8, swap>()); break; case VAR_BYTE: dst[i_dst] = ReadNormalized<T, s8>(src.Read<s8, swap>()); break; case VAR_UNSIGNED_SHORT: dst[i_dst] = ReadNormalized<T, u16>(src.Read<u16, swap>()); break; case VAR_SHORT: dst[i_dst] = ReadNormalized<T, s16>(src.Read<s16, swap>()); break; case VAR_FLOAT: dst[i_dst] = ReadNormalized<T, float>(src.Read<float, swap>()); break; } _assert_msg_(VIDEO, !format.integer || format.type != VAR_FLOAT, "only non-float values are allowed to be streamed as integer"); } for (; i < components; i++) { int i_dst = reverse ? components - i - 1 : i; dst[i_dst] = i == 3; } } } void SWVertexLoader::ParseVertex(const PortableVertexDeclaration& vdec, int index) { DataReader src(LocalVBuffer.data(), LocalVBuffer.data() + LocalVBuffer.size()); src.Skip(index * vdec.stride); ReadVertexAttribute<float>(&m_Vertex.position[0], src, vdec.position, 0, 3, false); for (int i = 0; i < 3; i++) { ReadVertexAttribute<float>(&m_Vertex.normal[i][0], src, vdec.normals[i], 0, 3, false); } for (int i = 0; i < 2; i++) { ReadVertexAttribute<u8>(m_Vertex.color[i], src, vdec.colors[i], 0, 4, true); } for (int i = 0; i < 8; i++) { ReadVertexAttribute<float>(m_Vertex.texCoords[i], src, vdec.texcoords[i], 0, 2, false); // the texmtr is stored as third component of the texCoord if (vdec.texcoords[i].components >= 3) { ReadVertexAttribute<u8>(&m_Vertex.texMtx[i], src, vdec.texcoords[i], 2, 1, false); } } ReadVertexAttribute<u8>(&m_Vertex.posMtx, src, vdec.posmtx, 0, 1, false); }