mirror of
				https://github.com/PabloMK7/citra.git
				synced 2025-10-30 21:30:04 +00:00 
			
		
		
		
	Merge pull request #2721 from wwylele/texture-cube
swrasterizer: implemented TextureCube
This commit is contained in:
		
						commit
						54ea95cca7
					
				
					 2 changed files with 77 additions and 3 deletions
				
			
		|  | @ -133,7 +133,32 @@ struct TexturingRegs { | ||||||
|         BitField<16, 1, u32> clear_texture_cache; // TODO: unimplemented
 |         BitField<16, 1, u32> clear_texture_cache; // TODO: unimplemented
 | ||||||
|     } main_config; |     } main_config; | ||||||
|     TextureConfig texture0; |     TextureConfig texture0; | ||||||
|     INSERT_PADDING_WORDS(0x8); | 
 | ||||||
|  |     enum class CubeFace { | ||||||
|  |         PositiveX = 0, | ||||||
|  |         NegativeX = 1, | ||||||
|  |         PositiveY = 2, | ||||||
|  |         NegativeY = 3, | ||||||
|  |         PositiveZ = 4, | ||||||
|  |         NegativeZ = 5, | ||||||
|  |     }; | ||||||
|  | 
 | ||||||
|  |     BitField<0, 22, u32> cube_address[5]; | ||||||
|  | 
 | ||||||
|  |     PAddr GetCubePhysicalAddress(CubeFace face) const { | ||||||
|  |         PAddr address = texture0.address; | ||||||
|  |         if (face != CubeFace::PositiveX) { | ||||||
|  |             // Bits [22:27] from the main texture address is shared with all cubemap additional
 | ||||||
|  |             // addresses.
 | ||||||
|  |             auto& face_addr = cube_address[static_cast<size_t>(face) - 1]; | ||||||
|  |             address &= ~face_addr.mask; | ||||||
|  |             address |= face_addr; | ||||||
|  |         } | ||||||
|  |         // A multiplier of 8 is also needed in the same way as the main address.
 | ||||||
|  |         return address * 8; | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     INSERT_PADDING_WORDS(0x3); | ||||||
|     BitField<0, 4, TextureFormat> texture0_format; |     BitField<0, 4, TextureFormat> texture0_format; | ||||||
|     BitField<0, 1, u32> fragment_lighting_enable; |     BitField<0, 1, u32> fragment_lighting_enable; | ||||||
|     INSERT_PADDING_WORDS(0x1); |     INSERT_PADDING_WORDS(0x1); | ||||||
|  |  | ||||||
|  | @ -5,6 +5,7 @@ | ||||||
| #include <algorithm> | #include <algorithm> | ||||||
| #include <array> | #include <array> | ||||||
| #include <cmath> | #include <cmath> | ||||||
|  | #include <tuple> | ||||||
| #include "common/assert.h" | #include "common/assert.h" | ||||||
| #include "common/bit_field.h" | #include "common/bit_field.h" | ||||||
| #include "common/color.h" | #include "common/color.h" | ||||||
|  | @ -70,6 +71,49 @@ static int SignedArea(const Math::Vec2<Fix12P4>& vtx1, const Math::Vec2<Fix12P4> | ||||||
|     return Math::Cross(vec1, vec2).z; |     return Math::Cross(vec1, vec2).z; | ||||||
| }; | }; | ||||||
| 
 | 
 | ||||||
|  | /// Convert a 3D vector for cube map coordinates to 2D texture coordinates along with the face name
 | ||||||
|  | static std::tuple<float24, float24, PAddr> ConvertCubeCoord(float24 u, float24 v, float24 w, | ||||||
|  |                                                             const TexturingRegs& regs) { | ||||||
|  |     const float abs_u = std::abs(u.ToFloat32()); | ||||||
|  |     const float abs_v = std::abs(v.ToFloat32()); | ||||||
|  |     const float abs_w = std::abs(w.ToFloat32()); | ||||||
|  |     float24 x, y, z; | ||||||
|  |     PAddr addr; | ||||||
|  |     if (abs_u > abs_v && abs_u > abs_w) { | ||||||
|  |         if (u > float24::FromFloat32(0)) { | ||||||
|  |             addr = regs.GetCubePhysicalAddress(TexturingRegs::CubeFace::PositiveX); | ||||||
|  |             y = -v; | ||||||
|  |         } else { | ||||||
|  |             addr = regs.GetCubePhysicalAddress(TexturingRegs::CubeFace::NegativeX); | ||||||
|  |             y = v; | ||||||
|  |         } | ||||||
|  |         x = -w; | ||||||
|  |         z = u; | ||||||
|  |     } else if (abs_v > abs_w) { | ||||||
|  |         if (v > float24::FromFloat32(0)) { | ||||||
|  |             addr = regs.GetCubePhysicalAddress(TexturingRegs::CubeFace::PositiveY); | ||||||
|  |             x = u; | ||||||
|  |         } else { | ||||||
|  |             addr = regs.GetCubePhysicalAddress(TexturingRegs::CubeFace::NegativeY); | ||||||
|  |             x = -u; | ||||||
|  |         } | ||||||
|  |         y = w; | ||||||
|  |         z = v; | ||||||
|  |     } else { | ||||||
|  |         if (w > float24::FromFloat32(0)) { | ||||||
|  |             addr = regs.GetCubePhysicalAddress(TexturingRegs::CubeFace::PositiveZ); | ||||||
|  |             y = -v; | ||||||
|  |         } else { | ||||||
|  |             addr = regs.GetCubePhysicalAddress(TexturingRegs::CubeFace::NegativeZ); | ||||||
|  |             y = v; | ||||||
|  |         } | ||||||
|  |         x = u; | ||||||
|  |         z = w; | ||||||
|  |     } | ||||||
|  |     const float24 half = float24::FromFloat32(0.5f); | ||||||
|  |     return std::make_tuple(x / z * half + half, y / z * half + half, addr); | ||||||
|  | } | ||||||
|  | 
 | ||||||
| MICROPROFILE_DEFINE(GPU_Rasterization, "GPU", "Rasterization", MP_RGB(50, 50, 240)); | MICROPROFILE_DEFINE(GPU_Rasterization, "GPU", "Rasterization", MP_RGB(50, 50, 240)); | ||||||
| 
 | 
 | ||||||
| /**
 | /**
 | ||||||
|  | @ -284,10 +328,16 @@ static void ProcessTriangleInternal(const Vertex& v0, const Vertex& v1, const Ve | ||||||
| 
 | 
 | ||||||
|                 // Only unit 0 respects the texturing type (according to 3DBrew)
 |                 // Only unit 0 respects the texturing type (according to 3DBrew)
 | ||||||
|                 // TODO: Refactor so cubemaps and shadowmaps can be handled
 |                 // TODO: Refactor so cubemaps and shadowmaps can be handled
 | ||||||
|  |                 PAddr texture_address = texture.config.GetPhysicalAddress(); | ||||||
|                 if (i == 0) { |                 if (i == 0) { | ||||||
|                     switch (texture.config.type) { |                     switch (texture.config.type) { | ||||||
|                     case TexturingRegs::TextureConfig::Texture2D: |                     case TexturingRegs::TextureConfig::Texture2D: | ||||||
|                         break; |                         break; | ||||||
|  |                     case TexturingRegs::TextureConfig::TextureCube: { | ||||||
|  |                         auto w = GetInterpolatedAttribute(v0.tc0_w, v1.tc0_w, v2.tc0_w); | ||||||
|  |                         std::tie(u, v, texture_address) = ConvertCubeCoord(u, v, w, regs.texturing); | ||||||
|  |                         break; | ||||||
|  |                     } | ||||||
|                     case TexturingRegs::TextureConfig::Projection2D: { |                     case TexturingRegs::TextureConfig::Projection2D: { | ||||||
|                         auto tc0_w = GetInterpolatedAttribute(v0.tc0_w, v1.tc0_w, v2.tc0_w); |                         auto tc0_w = GetInterpolatedAttribute(v0.tc0_w, v1.tc0_w, v2.tc0_w); | ||||||
|                         u /= tc0_w; |                         u /= tc0_w; | ||||||
|  | @ -322,8 +372,7 @@ static void ProcessTriangleInternal(const Vertex& v0, const Vertex& v1, const Ve | ||||||
|                     t = texture.config.height - 1 - |                     t = texture.config.height - 1 - | ||||||
|                         GetWrappedTexCoord(texture.config.wrap_t, t, texture.config.height); |                         GetWrappedTexCoord(texture.config.wrap_t, t, texture.config.height); | ||||||
| 
 | 
 | ||||||
|                     u8* texture_data = |                     const u8* texture_data = Memory::GetPhysicalPointer(texture_address); | ||||||
|                         Memory::GetPhysicalPointer(texture.config.GetPhysicalAddress()); |  | ||||||
|                     auto info = |                     auto info = | ||||||
|                         Texture::TextureInfo::FromPicaRegister(texture.config, texture.format); |                         Texture::TextureInfo::FromPicaRegister(texture.config, texture.format); | ||||||
| 
 | 
 | ||||||
|  |  | ||||||
		Loading…
	
	Add table
		Add a link
		
	
		Reference in a new issue