mirror of
https://github.com/dolphin-emu/dolphin.git
synced 2025-01-24 15:01:16 +01:00
937bb2aa2e
Fixes LIT (https://bugs.dolphin-emu.org/issues/13635). The text does not include normals, but has lighting enabled. With the previous default of (0, 0, 0), lighting was always black (as dot(X, (0, 0, 0)) is always 0). It seems like the normal from the map in the background (0, 0, 1) is re-used. LIT also has the vertex color enabled while vertex color is not specified, the same as SMS's debug cubes; the default MissingColorValue GameINI value of solid white seems to work correctly in this case.
109 lines
3.7 KiB
C++
109 lines
3.7 KiB
C++
// Copyright 2008 Dolphin Emulator Project
|
|
// SPDX-License-Identifier: GPL-2.0-or-later
|
|
|
|
#pragma once
|
|
|
|
#include <array>
|
|
#include <memory>
|
|
#include <string>
|
|
#include <unordered_map>
|
|
|
|
#include "Common/CommonTypes.h"
|
|
#include "Common/EnumMap.h"
|
|
#include "VideoCommon/CPMemory.h"
|
|
|
|
class NativeVertexFormat;
|
|
struct PortableVertexDeclaration;
|
|
|
|
namespace OpcodeDecoder
|
|
{
|
|
enum class Primitive : u8;
|
|
}
|
|
|
|
namespace VertexLoaderManager
|
|
{
|
|
using NativeVertexFormatMap =
|
|
std::unordered_map<PortableVertexDeclaration, std::unique_ptr<NativeVertexFormat>>;
|
|
|
|
void Init();
|
|
void Clear();
|
|
|
|
void MarkAllDirty();
|
|
|
|
// Creates or obtains a pointer to a VertexFormat representing decl.
|
|
// If this results in a VertexFormat being created, if the game later uses a matching vertex
|
|
// declaration, the one that was previously created will be used.
|
|
NativeVertexFormat* GetOrCreateMatchingFormat(const PortableVertexDeclaration& decl);
|
|
|
|
// For vertex ubershaders, all attributes need to be present, even when the vertex
|
|
// format does not contain them. This function returns a vertex format with dummy
|
|
// offsets set to the unused attributes.
|
|
NativeVertexFormat* GetUberVertexFormat(const PortableVertexDeclaration& decl);
|
|
|
|
// Returns -1 if buf_size is insufficient, else the amount of bytes consumed
|
|
template <bool IsPreprocess = false>
|
|
int RunVertices(int vtx_attr_group, OpcodeDecoder::Primitive primitive, int count, const u8* src);
|
|
|
|
namespace detail
|
|
{
|
|
// This will look for an existing loader in the global hashmap or create a new one if there is none.
|
|
// It should not be used directly because RefreshLoaders() has another cache for fast lookups.
|
|
template <bool IsPreprocess = false>
|
|
VertexLoaderBase* GetOrCreateLoader(int vtx_attr_group);
|
|
} // namespace detail
|
|
|
|
NativeVertexFormat* GetCurrentVertexFormat();
|
|
|
|
// Resolved pointers to array bases. Used by vertex loaders.
|
|
extern Common::EnumMap<u8*, CPArray::TexCoord7> cached_arraybases;
|
|
void UpdateVertexArrayPointers();
|
|
|
|
// Position cache for zfreeze (3 vertices, 4 floats each to allow SIMD overwrite).
|
|
// These arrays are in reverse order.
|
|
extern std::array<std::array<float, 4>, 3> position_cache;
|
|
extern std::array<u32, 3> position_matrix_index_cache;
|
|
// Needed for the game "LIT", which has text that has lighting enabled, but doesn't have normal
|
|
// vectors. The normals from the last drawn object are used instead.
|
|
// See https://bugs.dolphin-emu.org/issues/13635
|
|
extern std::array<float, 4> normal_cache;
|
|
// Store the tangent and binormal vectors for games that use emboss texgens when the vertex format
|
|
// doesn't include them (e.g. RS2 and RS3). These too are 4 floats each for SIMD overwrites.
|
|
extern std::array<float, 4> tangent_cache;
|
|
extern std::array<float, 4> binormal_cache;
|
|
|
|
// VB_HAS_X. Bitmask telling what vertex components are present.
|
|
extern u32 g_current_components;
|
|
|
|
extern BitSet8 g_main_vat_dirty;
|
|
extern BitSet8 g_preprocess_vat_dirty;
|
|
extern bool g_bases_dirty; // Main only
|
|
extern std::array<VertexLoaderBase*, CP_NUM_VAT_REG> g_main_vertex_loaders;
|
|
extern std::array<VertexLoaderBase*, CP_NUM_VAT_REG> g_preprocess_vertex_loaders;
|
|
extern bool g_needs_cp_xf_consistency_check;
|
|
|
|
template <bool IsPreprocess = false>
|
|
VertexLoaderBase* RefreshLoader(int vtx_attr_group)
|
|
{
|
|
constexpr const BitSet8& attr_dirty = IsPreprocess ? g_preprocess_vat_dirty : g_main_vat_dirty;
|
|
constexpr const auto& vertex_loaders =
|
|
IsPreprocess ? g_preprocess_vertex_loaders : g_main_vertex_loaders;
|
|
|
|
VertexLoaderBase* loader;
|
|
if (!attr_dirty[vtx_attr_group]) [[likely]]
|
|
{
|
|
loader = vertex_loaders[vtx_attr_group];
|
|
}
|
|
else [[unlikely]]
|
|
{
|
|
loader = detail::GetOrCreateLoader<IsPreprocess>(vtx_attr_group);
|
|
}
|
|
|
|
// Lookup pointers for any vertex arrays.
|
|
if constexpr (!IsPreprocess)
|
|
UpdateVertexArrayPointers();
|
|
|
|
return loader;
|
|
}
|
|
|
|
} // namespace VertexLoaderManager
|