2012-08-06 23:09:43 +02:00
|
|
|
// Copyright (C) 2003 Dolphin Project.
|
|
|
|
|
|
|
|
// This program is free software: you can redistribute it and/or modify
|
|
|
|
// it under the terms of the GNU General Public License as published by
|
|
|
|
// the Free Software Foundation, version 2.0.
|
|
|
|
|
|
|
|
// This program is distributed in the hope that it will be useful,
|
|
|
|
// but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
|
|
// MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
|
|
|
// GNU General Public License 2.0 for more details.
|
|
|
|
|
|
|
|
// A copy of the GPL 2.0 should have been included with the program.
|
|
|
|
// If not, see http://www.gnu.org/licenses/
|
|
|
|
|
|
|
|
// Official SVN repository and contact information can be found at
|
|
|
|
// http://code.google.com/p/dolphin-emu/
|
|
|
|
|
|
|
|
#ifndef _SHADERGENCOMMON_H
|
|
|
|
#define _SHADERGENCOMMON_H
|
|
|
|
|
|
|
|
#include <stdio.h>
|
|
|
|
#include <stdarg.h>
|
2013-04-10 14:25:18 +02:00
|
|
|
#include <string>
|
2012-09-02 20:00:15 +02:00
|
|
|
#include <vector>
|
2013-04-10 12:54:22 +02:00
|
|
|
#include <algorithm>
|
|
|
|
#include <typeinfo>
|
2012-09-02 20:00:15 +02:00
|
|
|
|
2013-03-31 20:55:57 +02:00
|
|
|
#include "CommonTypes.h"
|
|
|
|
#include "VideoCommon.h"
|
|
|
|
|
2013-03-26 23:03:10 +01:00
|
|
|
class ShaderGeneratorInterface
|
|
|
|
{
|
|
|
|
public:
|
|
|
|
void Write(const char* fmt, ...) {}
|
|
|
|
const char* GetBuffer() { return NULL; }
|
|
|
|
void SetBuffer(char* buffer) { }
|
|
|
|
inline void SetConstantsUsed(unsigned int first_index, unsigned int last_index) {}
|
2013-03-29 22:24:49 +01:00
|
|
|
|
|
|
|
template<class uid_data>
|
|
|
|
uid_data& GetUidData() { return *(uid_data*)NULL; }
|
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
|
|
|
|
{
|
|
|
|
return memcmp(this->values, obj.values, sizeof(values)) == 0;
|
|
|
|
}
|
|
|
|
|
2013-03-29 14:54:44 +01:00
|
|
|
bool operator != (const ShaderUid& obj) const
|
|
|
|
{
|
|
|
|
return memcmp(this->values, obj.values, sizeof(values)) != 0;
|
|
|
|
}
|
|
|
|
|
2012-08-06 23:09:43 +02:00
|
|
|
// TODO: Store last frame used and order by that? makes much more sense anyway...
|
|
|
|
bool operator < (const ShaderUid& obj) const
|
|
|
|
{
|
2012-08-07 01:02:04 +02:00
|
|
|
for (unsigned int i = 0; i < sizeof(uid_data) / sizeof(u32); ++i)
|
2012-08-06 23:09:43 +02:00
|
|
|
{
|
|
|
|
if (this->values[i] < obj.values[i])
|
|
|
|
return true;
|
|
|
|
else if (this->values[i] > obj.values[i])
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
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
|
|
|
|
|
|
|
private:
|
|
|
|
union
|
|
|
|
{
|
|
|
|
uid_data data;
|
|
|
|
u32 values[sizeof(uid_data) / sizeof(u32)];
|
|
|
|
};
|
|
|
|
};
|
|
|
|
|
2013-03-29 22:24:49 +01:00
|
|
|
class ShaderCode : public ShaderGeneratorInterface
|
2012-08-06 23:09:43 +02:00
|
|
|
{
|
|
|
|
public:
|
|
|
|
ShaderCode() : buf(NULL), write_ptr(NULL)
|
|
|
|
{
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
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-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-03-29 20:33:28 +01:00
|
|
|
return true;
|
|
|
|
// 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>
|
|
|
|
static void WriteRegister(T& object, API_TYPE ApiType, const char *prefix, const u32 num)
|
|
|
|
{
|
|
|
|
if (ApiType == API_OPENGL)
|
|
|
|
return; // Nothing to do here
|
|
|
|
|
|
|
|
object.Write(" : register(%s%d)", prefix, num);
|
|
|
|
}
|
|
|
|
|
|
|
|
template<class T>
|
|
|
|
static void WriteLocation(T& object, API_TYPE ApiType, bool using_ubos)
|
|
|
|
{
|
|
|
|
if (using_ubos)
|
|
|
|
return;
|
|
|
|
|
|
|
|
object.Write("uniform ");
|
|
|
|
}
|
|
|
|
|
|
|
|
template<class T>
|
|
|
|
static void DeclareUniform(T& object, API_TYPE api_type, bool using_ubos, const u32 num, const char* type, const char* name)
|
|
|
|
{
|
|
|
|
WriteLocation(object, api_type, using_ubos);
|
|
|
|
object.Write("%s %s ", type, name);
|
|
|
|
WriteRegister(object, api_type, "c", num);
|
|
|
|
object.Write(";\n");
|
|
|
|
}
|
|
|
|
|
2013-03-31 23:29:33 +02:00
|
|
|
struct LightingUidData
|
|
|
|
{
|
|
|
|
struct
|
|
|
|
{
|
|
|
|
u32 matsource : 1;
|
|
|
|
u32 enablelighting : 1;
|
|
|
|
u32 ambsource : 1;
|
|
|
|
u32 diffusefunc : 2;
|
|
|
|
u32 attnfunc : 2;
|
|
|
|
u32 light_mask : 8;
|
|
|
|
} lit_chans[4];
|
|
|
|
};
|
|
|
|
|
2013-04-10 12:54:22 +02:00
|
|
|
struct pixel_shader_uid_data;
|
|
|
|
struct vertex_shader_uid_data;
|
|
|
|
|
|
|
|
typedef ShaderUid<pixel_shader_uid_data> PixelShaderUid;
|
|
|
|
typedef ShaderUid<vertex_shader_uid_data> VertexShaderUid;
|
|
|
|
|
|
|
|
template<class UidT, class CodeT>
|
|
|
|
void CheckForUidMismatch(CodeT& new_code, const UidT& new_uid)
|
|
|
|
{
|
|
|
|
static std::map<UidT,std::string> s_shaders;
|
|
|
|
static std::vector<UidT> s_uids;
|
|
|
|
|
|
|
|
bool uid_is_indexed = std::find(s_uids.begin(), s_uids.end(), new_uid) != s_uids.end();
|
|
|
|
if (!uid_is_indexed)
|
|
|
|
{
|
|
|
|
s_uids.push_back(new_uid);
|
|
|
|
s_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 = s_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(),
|
|
|
|
(typeid(UidT) == typeid(PixelShaderUid)) ? "p" : (typeid(UidT) == typeid(VertexShaderUid)) ? "v" : "o",
|
|
|
|
++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.close();
|
|
|
|
|
|
|
|
// TODO: Make this more idiot-proof
|
|
|
|
ERROR_LOG(VIDEO, "%s shader uid mismatch!",
|
|
|
|
(typeid(UidT) == typeid(PixelShaderUid)) ? "Pixel" : (typeid(UidT) == typeid(VertexShaderUid)) ? "Vertex" : "Other");
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2012-08-06 23:09:43 +02:00
|
|
|
#endif // _SHADERGENCOMMON_H
|