// 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 #include #include #include #include #include #include "CommonTypes.h" #include "VideoCommon.h" 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) {} template uid_data& GetUidData() { return *(uid_data*)NULL; } }; template class ShaderUid : public ShaderGeneratorInterface { public: ShaderUid() { // TODO: Move to Shadergen => can be optimized out memset(values, 0, sizeof(values)); } bool operator == (const ShaderUid& obj) const { return memcmp(this->values, obj.values, sizeof(values)) == 0; } bool operator != (const ShaderUid& obj) const { return memcmp(this->values, obj.values, sizeof(values)) != 0; } // TODO: Store last frame used and order by that? makes much more sense anyway... bool operator < (const ShaderUid& obj) const { for (unsigned int i = 0; i < sizeof(uid_data) / sizeof(u32); ++i) { if (this->values[i] < obj.values[i]) return true; else if (this->values[i] > obj.values[i]) return false; } return false; } template inline T& GetUidData() { return data; } private: union { uid_data data; u32 values[sizeof(uid_data) / sizeof(u32)]; }; }; class ShaderCode : public ShaderGeneratorInterface { 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; }; class ShaderConstantProfile : public ShaderGeneratorInterface { 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) { return true; // return constant_usage[index]; } private: std::vector constant_usage; // TODO: Is vector appropriate here? }; template 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 static void WriteLocation(T& object, API_TYPE ApiType, bool using_ubos) { if (using_ubos) return; object.Write("uniform "); } template 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"); } 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]; }; struct pixel_shader_uid_data; struct vertex_shader_uid_data; typedef ShaderUid PixelShaderUid; typedef ShaderUid VertexShaderUid; template void CheckForUidMismatch(CodeT& new_code, const UidT& new_uid) { static std::map s_shaders; static std::vector 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"); } } } #endif // _SHADERGENCOMMON_H