#include "ppsspp_config.h"
#include <cstdint>
#include <cfloat>
#include <vector>
#include <string>
#include <d3d11.h>
#include <D3Dcompiler.h>
#if PPSSPP_PLATFORM(UWP)
#define ptr_D3DCompile D3DCompile
#else
#include "Common/GPU/D3D11/D3D11Loader.h"
#endif
#include "Common/CommonFuncs.h"
#include "Common/Log.h"
#include "Common/StringUtils.h"
#include "D3D11Util.h"
#include <wrl/client.h>
using namespace Microsoft::WRL;
std::vector<uint8_t> CompileShaderToBytecodeD3D11(const char *code, size_t codeSize, const char *target, UINT flags) {
ComPtr<ID3DBlob> compiledCode;
ComPtr<ID3DBlob> errorMsgs;
HRESULT result = ptr_D3DCompile(code, codeSize, nullptr, nullptr, nullptr, "main", target, flags, 0, &compiledCode, &errorMsgs);
std::string errors;
if (errorMsgs) {
errors = std::string((const char *)errorMsgs->GetBufferPointer(), errorMsgs->GetBufferSize() > 1 ? (errorMsgs->GetBufferSize() - 1) : 0);
std::string numberedCode = LineNumberString(code);
if (SUCCEEDED(result)) {
std::vector<std::string_view> lines;
SplitString(errors, '\n', lines);
for (auto &line : lines) {
auto trimmed = StripSpaces(line);
if (trimmed.find("pow(f, e) will not work for negative f") != std::string::npos) {
continue;
}
if (trimmed.size() > 1) {
WARN_LOG(Log::G3D, "%.*s", (int)trimmed.length(), trimmed.data());
}
}
} else {
ERROR_LOG(Log::G3D, "%s: %s\n\n%s", "errors", errors.c_str(), numberedCode.c_str());
OutputDebugStringA(errors.c_str());
OutputDebugStringA(numberedCode.c_str());
}
}
if (compiledCode) {
const uint8_t *buf = (const uint8_t *)compiledCode->GetBufferPointer();
std::vector<uint8_t> compiled = std::vector<uint8_t>(buf, buf + compiledCode->GetBufferSize());
_assert_(compiled.size() != 0);
return compiled;
}
return std::vector<uint8_t>();
}
HRESULT CreateVertexShaderD3D11(ID3D11Device *device, const char *code, size_t codeSize, std::vector<uint8_t> *byteCodeOut, D3D_FEATURE_LEVEL featureLevel, UINT flags, ID3D11VertexShader **ppVertexShader) {
if (ppVertexShader)
*ppVertexShader = nullptr;
const char *profile = featureLevel <= D3D_FEATURE_LEVEL_9_3 ? "vs_4_0_level_9_1" : "vs_4_0";
std::vector<uint8_t> byteCode = CompileShaderToBytecodeD3D11(code, codeSize, profile, flags);
if (byteCode.empty())
return S_FALSE;
auto hr = device->CreateVertexShader(byteCode.data(), byteCode.size(), nullptr, ppVertexShader);
if (byteCodeOut)
*byteCodeOut = byteCode;
return hr;
}
HRESULT CreatePixelShaderD3D11(ID3D11Device *device, const char *code, size_t codeSize, D3D_FEATURE_LEVEL featureLevel, UINT flags, ID3D11PixelShader **ppPixelShader) {
if (ppPixelShader)
*ppPixelShader = nullptr;
const char *profile = featureLevel <= D3D_FEATURE_LEVEL_9_3 ? "ps_4_0_level_9_1" : "ps_4_0";
std::vector<uint8_t> byteCode = CompileShaderToBytecodeD3D11(code, codeSize, profile, flags);
if (byteCode.empty())
return S_FALSE;
return device->CreatePixelShader(byteCode.data(), byteCode.size(), nullptr, ppPixelShader);
}
HRESULT CreateComputeShaderD3D11(ID3D11Device *device, const char *code, size_t codeSize, D3D_FEATURE_LEVEL featureLevel, UINT flags, ID3D11ComputeShader **ppComputeShader) {
if (ppComputeShader)
*ppComputeShader = nullptr;
if (featureLevel <= D3D_FEATURE_LEVEL_9_3)
return S_FALSE;
std::vector<uint8_t> byteCode = CompileShaderToBytecodeD3D11(code, codeSize, "cs_4_0", flags);
if (byteCode.empty())
return S_FALSE;
return device->CreateComputeShader(byteCode.data(), byteCode.size(), nullptr, ppComputeShader);
}
HRESULT CreateGeometryShaderD3D11(ID3D11Device *device, const char *code, size_t codeSize, D3D_FEATURE_LEVEL featureLevel, UINT flags, ID3D11GeometryShader **ppGeometryShader) {
if (ppGeometryShader)
*ppGeometryShader = nullptr;
if (featureLevel <= D3D_FEATURE_LEVEL_9_3)
return S_FALSE;
std::vector<uint8_t> byteCode = CompileShaderToBytecodeD3D11(code, codeSize, "gs_5_0", flags);
if (byteCode.empty())
return S_FALSE;
return device->CreateGeometryShader(byteCode.data(), byteCode.size(), nullptr, ppGeometryShader);
}