2015-05-24 06:55:12 +02:00
|
|
|
// Copyright 2012 Dolphin Emulator Project
|
2021-07-05 03:22:19 +02:00
|
|
|
// SPDX-License-Identifier: GPL-2.0-or-later
|
2012-08-06 23:09:43 +02:00
|
|
|
|
2014-02-10 13:54:46 -05:00
|
|
|
#pragma once
|
2012-08-06 23:09:43 +02:00
|
|
|
|
2016-01-17 16:54:31 -05:00
|
|
|
#include <cstring>
|
2022-02-18 06:34:01 -05:00
|
|
|
#include <functional>
|
2019-12-03 04:17:23 -05:00
|
|
|
#include <iterator>
|
2013-04-10 14:25:18 +02:00
|
|
|
#include <string>
|
2019-05-30 01:05:06 -04:00
|
|
|
#include <type_traits>
|
2012-09-02 20:00:15 +02:00
|
|
|
#include <vector>
|
|
|
|
|
2019-12-03 04:17:23 -05:00
|
|
|
#include <fmt/format.h>
|
|
|
|
|
2021-07-28 21:11:15 -07:00
|
|
|
#include "Common/BitField.h"
|
2014-02-17 05:18:15 -05:00
|
|
|
#include "Common/CommonTypes.h"
|
2021-11-18 12:56:25 -08:00
|
|
|
#include "Common/EnumMap.h"
|
2014-06-03 01:08:54 -04:00
|
|
|
#include "Common/StringUtil.h"
|
2021-07-26 11:20:04 -07:00
|
|
|
#include "Common/TypeUtils.h"
|
2019-12-22 13:40:40 -05:00
|
|
|
|
2021-11-18 12:56:25 -08:00
|
|
|
#include "VideoCommon/VideoCommon.h"
|
2013-03-31 20:55:57 +02:00
|
|
|
|
2013-04-25 13:30:41 +02:00
|
|
|
/**
|
2016-01-02 16:20:01 +10:00
|
|
|
* Common interface for classes that need to go through the shader generation path
|
|
|
|
* (GenerateVertexShader, GenerateGeometryShader, GeneratePixelShader)
|
2013-04-25 13:30:41 +02:00
|
|
|
* In particular, this includes the shader code generator (ShaderCode).
|
|
|
|
* A different class (ShaderUid) can be used to uniquely identify each ShaderCode object.
|
|
|
|
* More interesting things can be done with this, e.g. ShaderConstantProfile checks what shader
|
|
|
|
* constants are being used. This can be used to optimize buffer management.
|
2016-01-02 16:20:01 +10:00
|
|
|
* If the class does not use one or more of these methods (e.g. Uid class does not need code), the
|
|
|
|
* method will be defined as a no-op by the base class, and the call
|
|
|
|
* should be optimized out. The reason for this implementation is so that shader
|
|
|
|
* selection/generation can be done in two passes, with only a cache lookup being
|
|
|
|
* required if the shader has already been generated.
|
2013-04-25 13:30:41 +02:00
|
|
|
*/
|
2013-03-26 23:03:10 +01:00
|
|
|
class ShaderGeneratorInterface
|
|
|
|
{
|
|
|
|
public:
|
2013-04-25 13:30:41 +02:00
|
|
|
/*
|
|
|
|
* Used when the shader generator would write a piece of ShaderCode.
|
|
|
|
* Can be used like printf.
|
|
|
|
* @note In the ShaderCode implementation, this does indeed write the parameter string to an
|
|
|
|
* internal buffer. However, you're free to do whatever you like with the parameter.
|
|
|
|
*/
|
2016-01-02 16:20:01 +10:00
|
|
|
void Write(const char*, ...)
|
2015-10-16 18:00:44 -04:00
|
|
|
#ifdef __GNUC__
|
|
|
|
__attribute__((format(printf, 2, 3)))
|
|
|
|
#endif
|
2015-12-26 16:00:23 -05:00
|
|
|
{
|
|
|
|
}
|
2016-06-24 10:43:46 +02:00
|
|
|
|
2013-04-25 13:30:41 +02:00
|
|
|
/*
|
|
|
|
* Tells us that a specific constant range (including last_index) is being used by the shader
|
|
|
|
*/
|
2016-01-02 16:20:01 +10:00
|
|
|
void SetConstantsUsed(unsigned int first_index, unsigned int last_index) {}
|
2013-03-26 23:03:10 +01:00
|
|
|
};
|
|
|
|
|
2016-01-02 16:20:01 +10:00
|
|
|
/*
|
2013-04-25 13:30:41 +02:00
|
|
|
* Shader UID class used to uniquely identify the ShaderCode output written in the shader generator.
|
|
|
|
* uid_data can be any struct of parameters that uniquely identify each shader code output.
|
|
|
|
* Unless performance is not an issue, uid_data should be tightly packed to reduce memory footprint.
|
|
|
|
* Shader generators will write to specific uid_data fields; ShaderUid methods will only read raw
|
|
|
|
* u32 values from a union.
|
2016-01-02 16:20:01 +10:00
|
|
|
* NOTE: Because LinearDiskCache reads and writes the storage associated with a ShaderUid instance,
|
|
|
|
* ShaderUid must be trivially copyable.
|
2013-04-25 13:30:41 +02:00
|
|
|
*/
|
2013-03-26 23:03:10 +01:00
|
|
|
template <class uid_data>
|
2013-03-29 22:24:49 +01:00
|
|
|
class ShaderUid : public ShaderGeneratorInterface
|
2012-08-06 23:09:43 +02:00
|
|
|
{
|
|
|
|
public:
|
2019-05-30 01:05:06 -04:00
|
|
|
static_assert(std::is_trivially_copyable_v<uid_data>,
|
|
|
|
"uid_data must be a trivially copyable type");
|
|
|
|
|
2022-06-04 22:37:06 -05:00
|
|
|
ShaderUid() { memset(GetUidData(), 0, GetUidDataSize()); }
|
|
|
|
|
2012-08-06 23:09:43 +02:00
|
|
|
bool operator==(const ShaderUid& obj) const
|
|
|
|
{
|
2019-05-30 15:56:47 -04:00
|
|
|
return memcmp(GetUidData(), obj.GetUidData(), GetUidDataSize()) == 0;
|
2012-08-06 23:09:43 +02:00
|
|
|
}
|
2016-06-24 10:43:46 +02:00
|
|
|
|
2019-05-30 02:13:37 -04:00
|
|
|
bool operator!=(const ShaderUid& obj) const { return !operator==(obj); }
|
2016-06-24 10:43:46 +02:00
|
|
|
|
2013-04-25 13:30:41 +02:00
|
|
|
// determines the storage order inside STL containers
|
2012-08-06 23:09:43 +02:00
|
|
|
bool operator<(const ShaderUid& obj) const
|
|
|
|
{
|
2019-05-30 15:56:47 -04:00
|
|
|
return memcmp(GetUidData(), obj.GetUidData(), GetUidDataSize()) < 0;
|
2012-08-06 23:09:43 +02:00
|
|
|
}
|
2016-06-24 10:43:46 +02:00
|
|
|
|
2019-05-30 01:05:06 -04:00
|
|
|
// Returns a pointer to an internally stored object of the uid_data type.
|
|
|
|
uid_data* GetUidData() { return &data; }
|
|
|
|
|
|
|
|
// Returns a pointer to an internally stored object of the uid_data type.
|
2014-10-21 02:52:45 -04:00
|
|
|
const uid_data* GetUidData() const { return &data; }
|
2019-05-30 01:05:06 -04:00
|
|
|
|
|
|
|
// Returns the raw bytes that make up the shader UID.
|
2019-05-30 02:06:55 -04:00
|
|
|
const u8* GetUidDataRaw() const { return reinterpret_cast<const u8*>(&data); }
|
2019-05-30 01:05:06 -04:00
|
|
|
|
|
|
|
// Returns the size of the underlying UID data structure in bytes.
|
2019-05-30 02:06:55 -04:00
|
|
|
size_t GetUidDataSize() const { return sizeof(data); }
|
2018-04-12 14:18:04 +02:00
|
|
|
|
2012-08-06 23:09:43 +02:00
|
|
|
private:
|
2019-05-30 02:06:55 -04:00
|
|
|
uid_data data{};
|
2012-08-06 23:09:43 +02:00
|
|
|
};
|
|
|
|
|
2013-03-29 22:24:49 +01:00
|
|
|
class ShaderCode : public ShaderGeneratorInterface
|
2012-08-06 23:09:43 +02:00
|
|
|
{
|
|
|
|
public:
|
2015-12-26 16:00:23 -05:00
|
|
|
ShaderCode() { m_buffer.reserve(16384); }
|
2016-01-02 16:20:01 +10:00
|
|
|
const std::string& GetBuffer() const { return m_buffer; }
|
2019-12-03 04:17:23 -05:00
|
|
|
|
|
|
|
// Writes format strings using fmtlib format strings.
|
|
|
|
template <typename... Args>
|
2022-01-12 16:59:19 -08:00
|
|
|
void Write(fmt::format_string<Args...> format, Args&&... args)
|
2019-12-03 04:17:23 -05:00
|
|
|
{
|
|
|
|
fmt::format_to(std::back_inserter(m_buffer), format, std::forward<Args>(args)...);
|
|
|
|
}
|
|
|
|
|
2016-01-02 16:20:01 +10:00
|
|
|
protected:
|
|
|
|
std::string m_buffer;
|
2012-08-06 23:09:43 +02:00
|
|
|
};
|
|
|
|
|
2013-04-25 13:30:41 +02:00
|
|
|
/**
|
|
|
|
* Generates a shader constant profile which can be used to query which constants are used in a
|
|
|
|
* shader
|
|
|
|
*/
|
2013-03-29 22:24:49 +01:00
|
|
|
class ShaderConstantProfile : public ShaderGeneratorInterface
|
2012-09-02 20:00:15 +02:00
|
|
|
{
|
|
|
|
public:
|
|
|
|
ShaderConstantProfile(int num_constants) { constant_usage.resize(num_constants); }
|
2016-01-02 16:20:01 +10:00
|
|
|
void SetConstantsUsed(unsigned int first_index, unsigned int last_index)
|
2012-09-02 20:00:15 +02:00
|
|
|
{
|
2015-02-15 14:43:31 -05:00
|
|
|
for (unsigned int i = first_index; i < last_index + 1; ++i)
|
2012-09-02 20:00:15 +02:00
|
|
|
constant_usage[i] = true;
|
|
|
|
}
|
2016-06-24 10:43:46 +02:00
|
|
|
|
2015-12-26 16:00:23 -05:00
|
|
|
bool ConstantIsUsed(unsigned int index) const
|
2012-09-02 20:00:15 +02:00
|
|
|
{
|
2013-04-25 13:30:41 +02:00
|
|
|
// TODO: Not ready for usage yet
|
2013-03-29 20:33:28 +01:00
|
|
|
return true;
|
|
|
|
// return constant_usage[index];
|
2012-09-02 20:00:15 +02:00
|
|
|
}
|
2016-01-02 16:20:01 +10:00
|
|
|
|
2012-09-02 20:00:15 +02:00
|
|
|
private:
|
|
|
|
std::vector<bool> constant_usage; // TODO: Is vector<bool> appropriate here?
|
|
|
|
};
|
2014-05-30 16:17:30 +02:00
|
|
|
|
2017-07-20 17:10:02 +10:00
|
|
|
// Host config contains the settings which can influence generated shaders.
|
|
|
|
union ShaderHostConfig
|
|
|
|
{
|
|
|
|
u32 bits;
|
|
|
|
|
2021-07-28 21:11:15 -07:00
|
|
|
BitField<0, 1, bool, u32> msaa;
|
|
|
|
BitField<1, 1, bool, u32> ssaa;
|
|
|
|
BitField<2, 1, bool, u32> stereo;
|
|
|
|
BitField<3, 1, bool, u32> wireframe;
|
|
|
|
BitField<4, 1, bool, u32> per_pixel_lighting;
|
|
|
|
BitField<5, 1, bool, u32> vertex_rounding;
|
|
|
|
BitField<6, 1, bool, u32> fast_depth_calc;
|
|
|
|
BitField<7, 1, bool, u32> bounding_box;
|
|
|
|
BitField<8, 1, bool, u32> backend_dual_source_blend;
|
|
|
|
BitField<9, 1, bool, u32> backend_geometry_shaders;
|
|
|
|
BitField<10, 1, bool, u32> backend_early_z;
|
|
|
|
BitField<11, 1, bool, u32> backend_bbox;
|
|
|
|
BitField<12, 1, bool, u32> backend_gs_instancing;
|
|
|
|
BitField<13, 1, bool, u32> backend_clip_control;
|
|
|
|
BitField<14, 1, bool, u32> backend_ssaa;
|
|
|
|
BitField<15, 1, bool, u32> backend_atomics;
|
|
|
|
BitField<16, 1, bool, u32> backend_depth_clamp;
|
|
|
|
BitField<17, 1, bool, u32> backend_reversed_depth_range;
|
|
|
|
BitField<18, 1, bool, u32> backend_bitfield;
|
|
|
|
BitField<19, 1, bool, u32> backend_dynamic_sampler_indexing;
|
|
|
|
BitField<20, 1, bool, u32> backend_shader_framebuffer_fetch;
|
|
|
|
BitField<21, 1, bool, u32> backend_logic_op;
|
|
|
|
BitField<22, 1, bool, u32> backend_palette_conversion;
|
2021-07-28 21:18:18 -07:00
|
|
|
BitField<23, 1, bool, u32> enable_validation_layer;
|
2021-07-29 17:43:35 -07:00
|
|
|
BitField<24, 1, bool, u32> manual_texture_sampling;
|
2021-08-01 17:31:40 -07:00
|
|
|
BitField<25, 1, bool, u32> manual_texture_sampling_custom_texture_sizes;
|
2021-08-06 23:28:26 -04:00
|
|
|
BitField<26, 1, bool, u32> backend_sampler_lod_bias;
|
2017-07-20 17:10:02 +10:00
|
|
|
|
|
|
|
static ShaderHostConfig GetCurrent();
|
|
|
|
};
|
|
|
|
|
|
|
|
// Gets the filename of the specified type of cache object (e.g. vertex shader, pipeline).
|
|
|
|
std::string GetDiskShaderCacheFileName(APIType api_type, const char* type, bool include_gameid,
|
2018-02-25 01:15:35 +10:00
|
|
|
bool include_host_config, bool include_api = true);
|
2017-07-20 17:10:02 +10:00
|
|
|
|
2021-09-07 17:27:18 -07:00
|
|
|
void WriteIsNanHeader(ShaderCode& out, APIType api_type);
|
2021-07-26 11:20:04 -07:00
|
|
|
void WriteBitfieldExtractHeader(ShaderCode& out, APIType api_type,
|
|
|
|
const ShaderHostConfig& host_config);
|
2021-09-07 17:27:18 -07:00
|
|
|
|
2019-12-22 13:40:40 -05:00
|
|
|
void GenerateVSOutputMembers(ShaderCode& object, APIType api_type, u32 texgens,
|
|
|
|
const ShaderHostConfig& host_config, std::string_view qualifier);
|
2016-08-04 14:09:35 +02:00
|
|
|
|
2019-12-22 13:40:40 -05:00
|
|
|
void AssignVSOutputMembers(ShaderCode& object, std::string_view a, std::string_view b, u32 texgens,
|
|
|
|
const ShaderHostConfig& host_config);
|
2014-11-02 23:40:52 +01:00
|
|
|
|
2015-09-06 13:58:18 +02:00
|
|
|
// We use the flag "centroid" to fix some MSAA rendering bugs. With MSAA, the
|
|
|
|
// pixel shader will be executed for each pixel which has at least one passed sample.
|
|
|
|
// So there may be rendered pixels where the center of the pixel isn't in the primitive.
|
|
|
|
// As the pixel shader usually renders at the center of the pixel, this position may be
|
|
|
|
// outside the primitive. This will lead to sampling outside the texture, sign changes, ...
|
|
|
|
// As a workaround, we interpolate at the centroid of the coveraged pixel, which
|
|
|
|
// is always inside the primitive.
|
|
|
|
// Without MSAA, this flag is defined to have no effect.
|
2019-12-22 13:40:40 -05:00
|
|
|
const char* GetInterpolationQualifier(bool msaa, bool ssaa, bool in_glsl_interface_block = false,
|
|
|
|
bool in = false);
|
2015-09-06 13:58:18 +02:00
|
|
|
|
2021-07-26 11:20:04 -07:00
|
|
|
// bitfieldExtract generator for BitField types
|
|
|
|
template <auto ptr_to_bitfield_member>
|
|
|
|
std::string BitfieldExtract(std::string_view source)
|
|
|
|
{
|
|
|
|
using BitFieldT = Common::MemberType<ptr_to_bitfield_member>;
|
2021-08-26 13:58:45 -07:00
|
|
|
return fmt::format("bitfieldExtract({}({}), {}, {})", BitFieldT::IsSigned() ? "int" : "uint",
|
|
|
|
source, static_cast<u32>(BitFieldT::StartBit()),
|
2021-07-26 11:20:04 -07:00
|
|
|
static_cast<u32>(BitFieldT::NumBits()));
|
|
|
|
}
|
|
|
|
|
2021-11-18 12:56:25 -08:00
|
|
|
template <auto last_member, typename = decltype(last_member)>
|
|
|
|
void WriteSwitch(ShaderCode& out, APIType ApiType, std::string_view variable,
|
|
|
|
const Common::EnumMap<std::string_view, last_member>& values, int indent,
|
|
|
|
bool break_)
|
|
|
|
{
|
|
|
|
const bool make_switch = (ApiType == APIType::D3D);
|
|
|
|
|
|
|
|
// The second template argument is needed to avoid compile errors from ambiguity with multiple
|
|
|
|
// enums with the same number of members in GCC prior to 8. See https://godbolt.org/z/xcKaW1seW
|
|
|
|
// and https://godbolt.org/z/hz7Yqq1P5
|
|
|
|
using enum_type = decltype(last_member);
|
|
|
|
|
|
|
|
// {:{}} is used to indent by formatting an empty string with a variable width
|
|
|
|
if (make_switch)
|
|
|
|
{
|
|
|
|
out.Write("{:{}}switch ({}) {{\n", "", indent, variable);
|
|
|
|
for (u32 i = 0; i <= static_cast<u32>(last_member); i++)
|
|
|
|
{
|
|
|
|
const enum_type key = static_cast<enum_type>(i);
|
|
|
|
|
|
|
|
// Assumes existence of an EnumFormatter
|
|
|
|
out.Write("{:{}}case {:s}:\n", "", indent, key);
|
|
|
|
// Note that this indentation behaves poorly for multi-line code
|
|
|
|
if (!values[key].empty())
|
|
|
|
out.Write("{:{}} {}\n", "", indent, values[key]);
|
|
|
|
if (break_)
|
|
|
|
out.Write("{:{}} break;\n", "", indent);
|
|
|
|
}
|
|
|
|
out.Write("{:{}}}}\n", "", indent);
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
// Generate a tree of if statements recursively
|
|
|
|
// std::function must be used because auto won't capture before initialization and thus can't be
|
|
|
|
// used recursively
|
|
|
|
std::function<void(u32, u32, u32)> BuildTree = [&](u32 cur_indent, u32 low, u32 high) {
|
|
|
|
// Each generated statement is for low <= x < high
|
|
|
|
if (high == low + 1)
|
|
|
|
{
|
|
|
|
// Down to 1 case (low <= x < low + 1 means x == low)
|
|
|
|
const enum_type key = static_cast<enum_type>(low);
|
|
|
|
// Note that this indentation behaves poorly for multi-line code
|
|
|
|
out.Write("{:{}}{} // {}\n", "", cur_indent, values[key], key);
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
u32 mid = low + ((high - low) / 2);
|
|
|
|
out.Write("{:{}}if ({} < {}u) {{\n", "", cur_indent, variable, mid);
|
|
|
|
BuildTree(cur_indent + 2, low, mid);
|
|
|
|
out.Write("{:{}}}} else {{\n", "", cur_indent);
|
|
|
|
BuildTree(cur_indent + 2, mid, high);
|
|
|
|
out.Write("{:{}}}}\n", "", cur_indent);
|
|
|
|
}
|
|
|
|
};
|
|
|
|
BuildTree(indent, 0, static_cast<u32>(last_member) + 1);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2014-05-30 16:17:30 +02:00
|
|
|
// Constant variable names
|
|
|
|
#define I_COLORS "color"
|
|
|
|
#define I_KCOLORS "k"
|
|
|
|
#define I_ALPHA "alphaRef"
|
|
|
|
#define I_TEXDIMS "texdim"
|
|
|
|
#define I_ZBIAS "czbias"
|
|
|
|
#define I_INDTEXSCALE "cindscale"
|
|
|
|
#define I_INDTEXMTX "cindmtx"
|
|
|
|
#define I_FOGCOLOR "cfogcolor"
|
|
|
|
#define I_FOGI "cfogi"
|
|
|
|
#define I_FOGF "cfogf"
|
2018-02-03 17:01:34 +10:00
|
|
|
#define I_FOGRANGE "cfogrange"
|
2014-12-25 00:34:22 -07:00
|
|
|
#define I_ZSLOPE "czslope"
|
2015-01-03 06:06:56 +13:00
|
|
|
#define I_EFBSCALE "cefbscale"
|
2016-06-24 10:43:46 +02:00
|
|
|
|
2014-05-30 16:17:30 +02:00
|
|
|
#define I_POSNORMALMATRIX "cpnmtx"
|
|
|
|
#define I_PROJECTION "cproj"
|
|
|
|
#define I_MATERIALS "cmtrl"
|
|
|
|
#define I_LIGHTS "clights"
|
|
|
|
#define I_TEXMATRICES "ctexmtx"
|
|
|
|
#define I_TRANSFORMMATRICES "ctrmtx"
|
|
|
|
#define I_NORMALMATRICES "cnmtx"
|
|
|
|
#define I_POSTTRANSFORMMATRICES "cpostmtx"
|
2014-10-25 02:59:02 +02:00
|
|
|
#define I_PIXELCENTERCORRECTION "cpixelcenter"
|
2017-02-18 18:22:41 -06:00
|
|
|
#define I_VIEWPORT_SIZE "cviewport"
|
2022-04-13 22:03:34 -07:00
|
|
|
#define I_CACHED_TANGENT "ctangent"
|
|
|
|
#define I_CACHED_BINORMAL "cbinormal"
|
2014-12-14 21:23:13 +01:00
|
|
|
|
2014-12-16 00:21:07 +01:00
|
|
|
#define I_STEREOPARAMS "cstereo"
|
|
|
|
#define I_LINEPTPARAMS "clinept"
|
|
|
|
#define I_TEXOFFSET "ctexoffset"
|
2016-06-24 10:43:46 +02:00
|
|
|
|
2017-07-20 15:25:27 +10:00
|
|
|
static const char s_shader_uniforms[] = "\tuint components;\n"
|
|
|
|
"\tuint xfmem_dualTexInfo;\n"
|
|
|
|
"\tuint xfmem_numColorChans;\n"
|
2021-02-22 19:03:06 -08:00
|
|
|
"\tuint missing_color_hex;\n"
|
|
|
|
"\tfloat4 missing_color_value;\n"
|
2017-07-20 15:25:27 +10:00
|
|
|
"\tfloat4 " I_POSNORMALMATRIX "[6];\n"
|
2014-10-29 14:15:12 +01:00
|
|
|
"\tfloat4 " I_PROJECTION "[4];\n"
|
|
|
|
"\tint4 " I_MATERIALS "[4];\n"
|
|
|
|
"\tLight " I_LIGHTS "[8];\n"
|
|
|
|
"\tfloat4 " I_TEXMATRICES "[24];\n"
|
|
|
|
"\tfloat4 " I_TRANSFORMMATRICES "[64];\n"
|
|
|
|
"\tfloat4 " I_NORMALMATRICES "[32];\n"
|
|
|
|
"\tfloat4 " I_POSTTRANSFORMMATRICES "[64];\n"
|
2017-02-18 18:22:41 -06:00
|
|
|
"\tfloat4 " I_PIXELCENTERCORRECTION ";\n"
|
2017-07-20 15:25:27 +10:00
|
|
|
"\tfloat2 " I_VIEWPORT_SIZE ";\n"
|
|
|
|
"\tuint4 xfmem_pack1[8];\n"
|
2022-04-13 22:03:34 -07:00
|
|
|
"\tfloat4 " I_CACHED_TANGENT ";\n"
|
|
|
|
"\tfloat4 " I_CACHED_BINORMAL ";\n"
|
2017-07-20 15:25:27 +10:00
|
|
|
"\t#define xfmem_texMtxInfo(i) (xfmem_pack1[(i)].x)\n"
|
|
|
|
"\t#define xfmem_postMtxInfo(i) (xfmem_pack1[(i)].y)\n"
|
|
|
|
"\t#define xfmem_color(i) (xfmem_pack1[(i)].z)\n"
|
|
|
|
"\t#define xfmem_alpha(i) (xfmem_pack1[(i)].w)\n";
|