mirror of
				https://github.com/dolphin-emu/dolphin.git
				synced 2025-10-25 17:39:09 +00:00 
			
		
		
		
	
		
			
				
	
	
		
			235 lines
		
	
	
	
		
			7.4 KiB
		
	
	
	
		
			C++
		
	
	
	
	
	
			
		
		
	
	
			235 lines
		
	
	
	
		
			7.4 KiB
		
	
	
	
		
			C++
		
	
	
	
	
	
| // Copyright 2009 Dolphin Emulator Project
 | |
| // SPDX-License-Identifier: GPL-2.0-or-later
 | |
| 
 | |
| #pragma once
 | |
| 
 | |
| #include <array>
 | |
| 
 | |
| #include "Common/EnumMap.h"
 | |
| #include "VideoCommon/BPMemory.h"
 | |
| 
 | |
| class Tev
 | |
| {
 | |
|   struct TevColor
 | |
|   {
 | |
|     constexpr TevColor() = default;
 | |
|     constexpr explicit TevColor(s16 a_, s16 b_, s16 g_, s16 r_) : a(a_), b(b_), g(g_), r(r_) {}
 | |
| 
 | |
|     s16 a = 0;
 | |
|     s16 b = 0;
 | |
|     s16 g = 0;
 | |
|     s16 r = 0;
 | |
| 
 | |
|     constexpr static TevColor All(s16 value) { return TevColor(value, value, value, value); }
 | |
| 
 | |
|     constexpr s16& operator[](int index)
 | |
|     {
 | |
|       switch (index)
 | |
|       {
 | |
|       case ALP_C:
 | |
|         return a;
 | |
|       case BLU_C:
 | |
|         return b;
 | |
|       case GRN_C:
 | |
|         return g;
 | |
|       case RED_C:
 | |
|         return r;
 | |
|       default:
 | |
|         // invalid
 | |
|         return a;
 | |
|       }
 | |
|     }
 | |
|   };
 | |
| 
 | |
|   struct TevColorRef
 | |
|   {
 | |
|     constexpr explicit TevColorRef(const s16& r_, const s16& g_, const s16& b_)
 | |
|         : r(r_), g(g_), b(b_)
 | |
|     {
 | |
|     }
 | |
| 
 | |
|     const s16& r;
 | |
|     const s16& g;
 | |
|     const s16& b;
 | |
| 
 | |
|     constexpr static TevColorRef Color(const TevColor& color)
 | |
|     {
 | |
|       return TevColorRef(color.r, color.g, color.b);
 | |
|     }
 | |
|     constexpr static TevColorRef All(const s16& value) { return TevColorRef(value, value, value); }
 | |
|     constexpr static TevColorRef Alpha(const TevColor& color) { return All(color.a); }
 | |
|   };
 | |
| 
 | |
|   struct TevAlphaRef
 | |
|   {
 | |
|     constexpr explicit TevAlphaRef(const TevColor& color) : a(color.a) {}
 | |
|     constexpr explicit TevAlphaRef(const s16& a_) : a(a_) {}
 | |
| 
 | |
|     const s16& a;
 | |
|   };
 | |
| 
 | |
|   struct TevKonstRef
 | |
|   {
 | |
|     constexpr explicit TevKonstRef(const s16& a_, const s16& r_, const s16& g_, const s16& b_)
 | |
|         : a(a_), r(r_), g(g_), b(b_)
 | |
|     {
 | |
|     }
 | |
| 
 | |
|     const s16& a;
 | |
|     const s16& r;
 | |
|     const s16& g;
 | |
|     const s16& b;
 | |
| 
 | |
|     constexpr static TevKonstRef Value(const s16& value)
 | |
|     {
 | |
|       return TevKonstRef(value, value, value, value);
 | |
|     }
 | |
|     constexpr static TevKonstRef Konst(const s16& alpha, const TevColor& color)
 | |
|     {
 | |
|       return TevKonstRef(alpha, color.r, color.g, color.b);
 | |
|     }
 | |
|   };
 | |
| 
 | |
|   struct InputRegType
 | |
|   {
 | |
|     unsigned a : 8;
 | |
|     unsigned b : 8;
 | |
|     unsigned c : 8;
 | |
|     signed d : 11;
 | |
|   };
 | |
| 
 | |
|   struct TextureCoordinateType
 | |
|   {
 | |
|     signed s : 24;
 | |
|     signed t : 24;
 | |
|   };
 | |
| 
 | |
|   // color order: ABGR
 | |
|   Common::EnumMap<TevColor, TevOutput::Color2> Reg;
 | |
|   std::array<TevColor, 4> KonstantColors;
 | |
|   TevColor RawTexColor;
 | |
|   TevColor TexColor;
 | |
|   TevColor RasColor;
 | |
|   TevColor StageKonst;
 | |
| 
 | |
|   // Fixed constants, corresponding to KonstSel
 | |
|   static constexpr s16 V0 = 0;
 | |
|   static constexpr s16 V1_8 = 32;
 | |
|   static constexpr s16 V1_4 = 64;
 | |
|   static constexpr s16 V3_8 = 96;
 | |
|   static constexpr s16 V1_2 = 128;
 | |
|   static constexpr s16 V5_8 = 159;
 | |
|   static constexpr s16 V3_4 = 191;
 | |
|   static constexpr s16 V7_8 = 223;
 | |
|   static constexpr s16 V1 = 255;
 | |
| 
 | |
|   u8 AlphaBump = 0;
 | |
|   u8 IndirectTex[4][4]{};
 | |
|   TextureCoordinateType TexCoord{};
 | |
| 
 | |
|   const Common::EnumMap<TevColorRef, TevColorArg::Zero> m_ColorInputLUT{
 | |
|       TevColorRef::Color(Reg[TevOutput::Prev]),    // prev.rgb
 | |
|       TevColorRef::Alpha(Reg[TevOutput::Prev]),    // prev.aaa
 | |
|       TevColorRef::Color(Reg[TevOutput::Color0]),  // c0.rgb
 | |
|       TevColorRef::Alpha(Reg[TevOutput::Color0]),  // c0.aaa
 | |
|       TevColorRef::Color(Reg[TevOutput::Color1]),  // c1.rgb
 | |
|       TevColorRef::Alpha(Reg[TevOutput::Color1]),  // c1.aaa
 | |
|       TevColorRef::Color(Reg[TevOutput::Color2]),  // c2.rgb
 | |
|       TevColorRef::Alpha(Reg[TevOutput::Color2]),  // c2.aaa
 | |
|       TevColorRef::Color(TexColor),                // tex.rgb
 | |
|       TevColorRef::Alpha(TexColor),                // tex.aaa
 | |
|       TevColorRef::Color(RasColor),                // ras.rgb
 | |
|       TevColorRef::Alpha(RasColor),                // ras.aaa
 | |
|       TevColorRef::All(V1),                        // one
 | |
|       TevColorRef::All(V1_2),                      // half
 | |
|       TevColorRef::Color(StageKonst),              // konst
 | |
|       TevColorRef::All(V0),                        // zero
 | |
|   };
 | |
|   const Common::EnumMap<TevAlphaRef, TevAlphaArg::Zero> m_AlphaInputLUT{
 | |
|       TevAlphaRef(Reg[TevOutput::Prev]),    // prev
 | |
|       TevAlphaRef(Reg[TevOutput::Color0]),  // c0
 | |
|       TevAlphaRef(Reg[TevOutput::Color1]),  // c1
 | |
|       TevAlphaRef(Reg[TevOutput::Color2]),  // c2
 | |
|       TevAlphaRef(TexColor),                // tex
 | |
|       TevAlphaRef(RasColor),                // ras
 | |
|       TevAlphaRef(StageKonst),              // konst
 | |
|       TevAlphaRef(V0),                      // zero
 | |
|   };
 | |
|   const Common::EnumMap<TevKonstRef, KonstSel::K3_A> m_KonstLUT{
 | |
|       TevKonstRef::Value(V1),    // 1
 | |
|       TevKonstRef::Value(V7_8),  // 7/8
 | |
|       TevKonstRef::Value(V3_4),  // 3/4
 | |
|       TevKonstRef::Value(V5_8),  // 5/8
 | |
|       TevKonstRef::Value(V1_2),  // 1/2
 | |
|       TevKonstRef::Value(V3_8),  // 3/8
 | |
|       TevKonstRef::Value(V1_4),  // 1/4
 | |
|       TevKonstRef::Value(V1_8),  // 1/8
 | |
| 
 | |
|       // These are "invalid" values, not meant to be used. On hardware,
 | |
|       // they all output zero.
 | |
|       TevKonstRef::Value(V0), TevKonstRef::Value(V0), TevKonstRef::Value(V0),
 | |
|       TevKonstRef::Value(V0),
 | |
| 
 | |
|       // These values are valid for RGB only; they're invalid for alpha
 | |
|       TevKonstRef::Konst(V0, KonstantColors[0]),  // Konst 0 RGB
 | |
|       TevKonstRef::Konst(V0, KonstantColors[1]),  // Konst 1 RGB
 | |
|       TevKonstRef::Konst(V0, KonstantColors[2]),  // Konst 2 RGB
 | |
|       TevKonstRef::Konst(V0, KonstantColors[3]),  // Konst 3 RGB
 | |
| 
 | |
|       TevKonstRef::Value(KonstantColors[0].r),  // Konst 0 Red
 | |
|       TevKonstRef::Value(KonstantColors[1].r),  // Konst 1 Red
 | |
|       TevKonstRef::Value(KonstantColors[2].r),  // Konst 2 Red
 | |
|       TevKonstRef::Value(KonstantColors[3].r),  // Konst 3 Red
 | |
|       TevKonstRef::Value(KonstantColors[0].g),  // Konst 0 Green
 | |
|       TevKonstRef::Value(KonstantColors[1].g),  // Konst 1 Green
 | |
|       TevKonstRef::Value(KonstantColors[2].g),  // Konst 2 Green
 | |
|       TevKonstRef::Value(KonstantColors[3].g),  // Konst 3 Green
 | |
|       TevKonstRef::Value(KonstantColors[0].b),  // Konst 0 Blue
 | |
|       TevKonstRef::Value(KonstantColors[1].b),  // Konst 1 Blue
 | |
|       TevKonstRef::Value(KonstantColors[2].b),  // Konst 2 Blue
 | |
|       TevKonstRef::Value(KonstantColors[3].b),  // Konst 3 Blue
 | |
|       TevKonstRef::Value(KonstantColors[0].a),  // Konst 0 Alpha
 | |
|       TevKonstRef::Value(KonstantColors[1].a),  // Konst 1 Alpha
 | |
|       TevKonstRef::Value(KonstantColors[2].a),  // Konst 2 Alpha
 | |
|       TevKonstRef::Value(KonstantColors[3].a),  // Konst 3 Alpha
 | |
|   };
 | |
|   static constexpr Common::EnumMap<s16, TevBias::Compare> s_BiasLUT{0, 128, -128, 0};
 | |
|   static constexpr Common::EnumMap<u8, TevScale::Divide2> s_ScaleLShiftLUT{0, 1, 2, 0};
 | |
|   static constexpr Common::EnumMap<u8, TevScale::Divide2> s_ScaleRShiftLUT{0, 0, 0, 1};
 | |
| 
 | |
|   enum BufferBase
 | |
|   {
 | |
|     DIRECT = 0,
 | |
|     DIRECT_TFETCH = 16,
 | |
|     INDIRECT = 32
 | |
|   };
 | |
| 
 | |
|   void SetRasColor(RasColorChan colorChan, u32 swaptable);
 | |
| 
 | |
|   void DrawColorRegular(const TevStageCombiner::ColorCombiner& cc, const InputRegType inputs[4]);
 | |
|   void DrawColorCompare(const TevStageCombiner::ColorCombiner& cc, const InputRegType inputs[4]);
 | |
|   void DrawAlphaRegular(const TevStageCombiner::AlphaCombiner& ac, const InputRegType inputs[4]);
 | |
|   void DrawAlphaCompare(const TevStageCombiner::AlphaCombiner& ac, const InputRegType inputs[4]);
 | |
| 
 | |
|   void Indirect(unsigned int stageNum, s32 s, s32 t);
 | |
| 
 | |
| public:
 | |
|   s32 Position[3]{};
 | |
|   u8 Color[2][4]{};  // must be RGBA for correct swap table ordering
 | |
|   TextureCoordinateType Uv[8]{};
 | |
|   s32 IndirectLod[4]{};
 | |
|   bool IndirectLinear[4]{};
 | |
|   s32 TextureLod[16]{};
 | |
|   bool TextureLinear[16]{};
 | |
| 
 | |
|   enum
 | |
|   {
 | |
|     ALP_C,
 | |
|     BLU_C,
 | |
|     GRN_C,
 | |
|     RED_C
 | |
|   };
 | |
| 
 | |
|   void SetKonstColors();
 | |
|   void Draw();
 | |
| };
 |