2014-02-10 13:54:46 -05:00
|
|
|
// Copyright 2014 Dolphin Emulator Project
|
|
|
|
// Licensed under GPLv2
|
|
|
|
// Refer to the license.txt file included.
|
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
|
|
|
|
2014-02-17 05:18:15 -05:00
|
|
|
#include <algorithm>
|
|
|
|
#include <cstdarg>
|
|
|
|
#include <cstdio>
|
2014-02-23 03:45:46 +01:00
|
|
|
#include <iomanip>
|
2013-04-10 14:25:18 +02:00
|
|
|
#include <string>
|
2012-09-02 20:00:15 +02:00
|
|
|
#include <vector>
|
|
|
|
|
2014-02-17 05:18:15 -05:00
|
|
|
#include "Common/CommonTypes.h"
|
|
|
|
#include "VideoCommon/VideoCommon.h"
|
2013-03-31 20:55:57 +02:00
|
|
|
|
2013-04-25 13:30:41 +02:00
|
|
|
/**
|
|
|
|
* Common interface for classes that need to go through the shader generation path (GenerateVertexShader, GeneratePixelShader)
|
|
|
|
* 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.
|
|
|
|
* Each of the ShaderCode, ShaderUid and ShaderConstantProfile child classes only implement the subset of ShaderGeneratorInterface methods that are required for the specific tasks.
|
|
|
|
*/
|
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.
|
|
|
|
*/
|
2013-03-26 23:03:10 +01:00
|
|
|
void Write(const char* fmt, ...) {}
|
2013-04-25 13:30:41 +02:00
|
|
|
|
|
|
|
/*
|
|
|
|
* Returns a read pointer to the internal buffer.
|
|
|
|
* @note When implementing this method in a child class, you likely want to return the argument of the last SetBuffer call here
|
|
|
|
* @note SetBuffer() should be called before using GetBuffer().
|
|
|
|
*/
|
2014-03-09 21:14:26 +01:00
|
|
|
const char* GetBuffer() { return nullptr; }
|
2013-04-25 13:30:41 +02:00
|
|
|
|
|
|
|
/*
|
|
|
|
* Can be used to give the object a place to write to. This should be called before using Write().
|
|
|
|
* @param buffer pointer to a char buffer that the object can write to
|
|
|
|
*/
|
2013-03-26 23:03:10 +01:00
|
|
|
void SetBuffer(char* buffer) { }
|
2013-04-25 13:30:41 +02:00
|
|
|
|
|
|
|
/*
|
|
|
|
* Tells us that a specific constant range (including last_index) is being used by the shader
|
|
|
|
*/
|
2013-03-26 23:03:10 +01:00
|
|
|
inline void SetConstantsUsed(unsigned int first_index, unsigned int last_index) {}
|
2013-03-29 22:24:49 +01:00
|
|
|
|
2013-04-25 13:30:41 +02:00
|
|
|
/*
|
|
|
|
* Returns a pointer to an internally stored object of the uid_data type.
|
2014-03-09 21:14:26 +01:00
|
|
|
* @warning since most child classes use the default implementation you shouldn't access this directly without adding precautions against nullptr access (e.g. via adding a dummy structure, cf. the vertex/pixel shader generators)
|
2013-04-25 13:30:41 +02:00
|
|
|
*/
|
2013-03-29 22:24:49 +01:00
|
|
|
template<class uid_data>
|
2014-03-09 21:14:26 +01:00
|
|
|
uid_data& GetUidData() { return *(uid_data*)nullptr; }
|
2013-03-26 23:03:10 +01: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.
|
|
|
|
*/
|
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:
|
|
|
|
ShaderUid()
|
|
|
|
{
|
2012-08-07 14:36:56 +02:00
|
|
|
// TODO: Move to Shadergen => can be optimized out
|
2012-08-06 23:09:43 +02:00
|
|
|
memset(values, 0, sizeof(values));
|
|
|
|
}
|
|
|
|
|
|
|
|
bool operator == (const ShaderUid& obj) const
|
|
|
|
{
|
2013-09-11 17:34:23 +02:00
|
|
|
return memcmp(this->values, obj.values, data.NumValues() * sizeof(*values)) == 0;
|
2012-08-06 23:09:43 +02:00
|
|
|
}
|
|
|
|
|
2013-03-29 14:54:44 +01:00
|
|
|
bool operator != (const ShaderUid& obj) const
|
|
|
|
{
|
2013-09-11 17:34:23 +02:00
|
|
|
return memcmp(this->values, obj.values, data.NumValues() * sizeof(*values)) != 0;
|
2013-03-29 14:54:44 +01: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
|
|
|
|
{
|
2013-09-11 17:34:23 +02:00
|
|
|
return memcmp(this->values, obj.values, data.NumValues() * sizeof(*values)) < 0;
|
2012-08-06 23:09:43 +02:00
|
|
|
}
|
|
|
|
|
2013-03-29 22:24:49 +01:00
|
|
|
template<class T>
|
2013-03-29 22:29:37 +01:00
|
|
|
inline T& GetUidData() { return data; }
|
2012-08-06 23:09:43 +02:00
|
|
|
|
2013-04-29 21:00:39 +02:00
|
|
|
const uid_data& GetUidData() const { return data; }
|
|
|
|
size_t GetUidDataSize() const { return sizeof(values); }
|
|
|
|
|
2012-08-06 23:09:43 +02:00
|
|
|
private:
|
|
|
|
union
|
|
|
|
{
|
|
|
|
uid_data data;
|
2013-07-02 14:32:52 +02:00
|
|
|
u8 values[sizeof(uid_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:
|
2014-03-09 21:14:26 +01:00
|
|
|
ShaderCode() : buf(nullptr), write_ptr(nullptr)
|
2012-08-06 23:09:43 +02:00
|
|
|
{
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
void Write(const char* fmt, ...)
|
|
|
|
{
|
|
|
|
va_list arglist;
|
|
|
|
va_start(arglist, fmt);
|
|
|
|
write_ptr += vsprintf(write_ptr, fmt, arglist);
|
|
|
|
va_end(arglist);
|
|
|
|
}
|
|
|
|
|
|
|
|
const char* GetBuffer() { return buf; }
|
|
|
|
void SetBuffer(char* buffer) { buf = buffer; write_ptr = buffer; }
|
|
|
|
|
|
|
|
private:
|
|
|
|
const char* buf;
|
|
|
|
char* write_ptr;
|
|
|
|
};
|
|
|
|
|
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); }
|
|
|
|
|
|
|
|
inline void SetConstantsUsed(unsigned int first_index, unsigned int last_index)
|
|
|
|
{
|
|
|
|
for (unsigned int i = first_index; i < last_index+1; ++i)
|
|
|
|
constant_usage[i] = true;
|
|
|
|
}
|
|
|
|
|
|
|
|
inline bool ConstantIsUsed(unsigned int index)
|
|
|
|
{
|
2013-04-25 13:30:41 +02:00
|
|
|
// TODO: Not ready for usage yet
|
2013-03-29 20:33:28 +01:00
|
|
|
return true;
|
2014-02-16 23:51:41 -05:00
|
|
|
//return constant_usage[index];
|
2012-09-02 20:00:15 +02:00
|
|
|
}
|
|
|
|
private:
|
|
|
|
std::vector<bool> constant_usage; // TODO: Is vector<bool> appropriate here?
|
|
|
|
};
|
|
|
|
|
2013-03-31 20:55:57 +02:00
|
|
|
template<class T>
|
2013-08-12 12:52:28 +02:00
|
|
|
static inline void WriteRegister(T& object, API_TYPE ApiType, const char *prefix, const u32 num)
|
2013-03-31 20:55:57 +02:00
|
|
|
{
|
|
|
|
if (ApiType == API_OPENGL)
|
|
|
|
return; // Nothing to do here
|
|
|
|
|
|
|
|
object.Write(" : register(%s%d)", prefix, num);
|
|
|
|
}
|
|
|
|
|
|
|
|
template<class T>
|
2014-01-30 16:38:11 +01:00
|
|
|
static inline void WriteLocation(T& object, API_TYPE ApiType)
|
2013-03-31 20:55:57 +02:00
|
|
|
{
|
2014-01-30 16:38:11 +01:00
|
|
|
if (ApiType == API_OPENGL)
|
2013-03-31 20:55:57 +02:00
|
|
|
return;
|
|
|
|
|
|
|
|
object.Write("uniform ");
|
|
|
|
}
|
|
|
|
|
|
|
|
template<class T>
|
2014-01-30 16:38:11 +01:00
|
|
|
static inline void DeclareUniform(T& object, API_TYPE api_type, const u32 num, const char* type, const char* name)
|
2013-03-31 20:55:57 +02:00
|
|
|
{
|
2014-01-30 16:38:11 +01:00
|
|
|
WriteLocation(object, api_type);
|
2013-03-31 20:55:57 +02:00
|
|
|
object.Write("%s %s ", type, name);
|
|
|
|
WriteRegister(object, api_type, "c", num);
|
|
|
|
object.Write(";\n");
|
|
|
|
}
|
|
|
|
|
2013-04-25 13:30:41 +02:00
|
|
|
/**
|
|
|
|
* Checks if there has been
|
|
|
|
*/
|
2013-04-10 12:54:22 +02:00
|
|
|
template<class UidT, class CodeT>
|
2013-04-29 21:00:39 +02:00
|
|
|
class UidChecker
|
2013-04-10 12:54:22 +02:00
|
|
|
{
|
2013-04-29 21:00:39 +02:00
|
|
|
public:
|
|
|
|
void Invalidate()
|
2013-04-10 12:54:22 +02:00
|
|
|
{
|
2013-04-29 21:00:39 +02:00
|
|
|
m_shaders.clear();
|
|
|
|
m_uids.clear();
|
2013-04-10 12:54:22 +02:00
|
|
|
}
|
2013-04-29 21:00:39 +02:00
|
|
|
|
|
|
|
void AddToIndexAndCheck(CodeT& new_code, const UidT& new_uid, const char* shader_type, const char* dump_prefix)
|
2013-04-10 12:54:22 +02:00
|
|
|
{
|
2013-04-29 21:00:39 +02:00
|
|
|
bool uid_is_indexed = std::find(m_uids.begin(), m_uids.end(), new_uid) != m_uids.end();
|
|
|
|
if (!uid_is_indexed)
|
2013-04-10 12:54:22 +02:00
|
|
|
{
|
2013-04-29 21:00:39 +02:00
|
|
|
m_uids.push_back(new_uid);
|
|
|
|
m_shaders[new_uid] = new_code.GetBuffer();
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
// uid is already in the index => check if there's a shader with the same uid but different code
|
|
|
|
auto& old_code = m_shaders[new_uid];
|
|
|
|
if (strcmp(old_code.c_str(), new_code.GetBuffer()) != 0)
|
|
|
|
{
|
|
|
|
static int num_failures = 0;
|
|
|
|
|
|
|
|
char szTemp[MAX_PATH];
|
|
|
|
sprintf(szTemp, "%s%ssuid_mismatch_%04i.txt", File::GetUserPath(D_DUMP_IDX).c_str(),
|
|
|
|
dump_prefix,
|
|
|
|
++num_failures);
|
|
|
|
|
|
|
|
// TODO: Should also dump uids
|
|
|
|
std::ofstream file;
|
|
|
|
OpenFStream(file, szTemp, std::ios_base::out);
|
|
|
|
file << "Old shader code:\n" << old_code;
|
|
|
|
file << "\n\nNew shader code:\n" << new_code.GetBuffer();
|
|
|
|
file << "\n\nShader uid:\n";
|
|
|
|
for (unsigned int i = 0; i < new_uid.GetUidDataSize(); ++i)
|
|
|
|
{
|
|
|
|
u32 value = ((u32*)&new_uid.GetUidData())[i];
|
|
|
|
if ((i % 4) == 0)
|
|
|
|
{
|
2013-11-13 04:03:46 -05:00
|
|
|
auto last_value = (i+3 < new_uid.GetUidDataSize()-1) ? i+3 : new_uid.GetUidDataSize();
|
2013-04-29 21:00:39 +02:00
|
|
|
file << std::setfill(' ') << std::dec;
|
|
|
|
file << "Values " << std::setw(2) << i << " - " << last_value << ": ";
|
|
|
|
}
|
|
|
|
|
|
|
|
file << std::setw(8) << std::setfill('0') << std::hex << value << std::setw(1);
|
|
|
|
if ((i % 4) < 3)
|
|
|
|
file << ' ';
|
|
|
|
else
|
|
|
|
file << std::endl;
|
|
|
|
}
|
|
|
|
file.close();
|
|
|
|
|
|
|
|
ERROR_LOG(VIDEO, "%s shader uid mismatch! See %s for details", shader_type, szTemp);
|
|
|
|
}
|
2013-04-10 12:54:22 +02:00
|
|
|
}
|
|
|
|
}
|
2013-10-29 01:23:17 -04:00
|
|
|
|
2013-04-29 21:00:39 +02:00
|
|
|
private:
|
|
|
|
std::map<UidT,std::string> m_shaders;
|
|
|
|
std::vector<UidT> m_uids;
|
|
|
|
};
|