2018-03-22 20:01:57 +01:00
|
|
|
#include "dxbc_analysis.h"
|
2017-10-16 17:50:09 +02:00
|
|
|
#include "dxbc_compiler.h"
|
|
|
|
#include "dxbc_module.h"
|
|
|
|
|
|
|
|
namespace dxvk {
|
|
|
|
|
|
|
|
DxbcModule::DxbcModule(DxbcReader& reader)
|
|
|
|
: m_header(reader) {
|
|
|
|
for (uint32_t i = 0; i < m_header.numChunks(); i++) {
|
|
|
|
|
|
|
|
// The chunk tag is stored at the beginning of each chunk
|
|
|
|
auto chunkReader = reader.clone(m_header.chunkOffset(i));
|
|
|
|
auto tag = chunkReader.readTag();
|
|
|
|
|
|
|
|
// The chunk size follows right after the four-character
|
|
|
|
// code. This does not include the eight bytes that are
|
|
|
|
// consumed by the FourCC and chunk length entry.
|
2017-11-01 00:01:40 +01:00
|
|
|
auto chunkLength = chunkReader.readu32();
|
|
|
|
|
|
|
|
chunkReader = chunkReader.clone(8);
|
|
|
|
chunkReader = chunkReader.resize(chunkLength);
|
2017-10-16 17:50:09 +02:00
|
|
|
|
|
|
|
if ((tag == "SHDR") || (tag == "SHEX"))
|
|
|
|
m_shexChunk = new DxbcShex(chunkReader);
|
|
|
|
|
2017-11-01 00:01:40 +01:00
|
|
|
if ((tag == "ISGN"))
|
|
|
|
m_isgnChunk = new DxbcIsgn(chunkReader);
|
|
|
|
|
|
|
|
if ((tag == "OSGN"))
|
|
|
|
m_osgnChunk = new DxbcIsgn(chunkReader);
|
|
|
|
|
|
|
|
// if ((tag == "OSG5"))
|
|
|
|
// m_osgnChunk = new DxbcIsgn(chunkReader);
|
|
|
|
|
2017-10-16 17:50:09 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
DxbcModule::~DxbcModule() {
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2018-04-15 21:00:08 +02:00
|
|
|
Rc<DxvkShader> DxbcModule::compile(
|
|
|
|
const DxbcOptions& options,
|
|
|
|
const std::string& fileName) const {
|
2017-10-16 17:50:09 +02:00
|
|
|
if (m_shexChunk == nullptr)
|
|
|
|
throw DxvkError("DxbcModule::compile: No SHDR/SHEX chunk");
|
|
|
|
|
2018-03-23 01:04:04 +01:00
|
|
|
DxbcAnalysisInfo analysisInfo;
|
|
|
|
|
2018-03-22 20:01:57 +01:00
|
|
|
DxbcAnalyzer analyzer(options,
|
2018-03-23 01:04:04 +01:00
|
|
|
m_shexChunk->version(),
|
2018-03-23 19:48:07 +01:00
|
|
|
m_isgnChunk, m_osgnChunk,
|
2018-03-23 01:04:04 +01:00
|
|
|
analysisInfo);
|
2017-12-18 00:28:54 +01:00
|
|
|
|
2018-05-26 17:46:49 +02:00
|
|
|
this->runAnalyzer(analyzer, m_shexChunk->slice());
|
|
|
|
|
2018-04-15 21:00:08 +02:00
|
|
|
DxbcCompiler compiler(
|
|
|
|
fileName, options,
|
2017-12-07 16:29:34 +01:00
|
|
|
m_shexChunk->version(),
|
2018-03-23 01:04:04 +01:00
|
|
|
m_isgnChunk, m_osgnChunk,
|
|
|
|
analysisInfo);
|
2017-12-07 16:29:34 +01:00
|
|
|
|
2018-03-22 20:01:57 +01:00
|
|
|
this->runCompiler(compiler, m_shexChunk->slice());
|
|
|
|
|
|
|
|
return compiler.finalize();
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
void DxbcModule::runAnalyzer(
|
|
|
|
DxbcAnalyzer& analyzer,
|
|
|
|
DxbcCodeSlice slice) const {
|
2017-12-18 00:28:54 +01:00
|
|
|
DxbcDecodeContext decoder;
|
|
|
|
|
|
|
|
while (!slice.atEnd()) {
|
|
|
|
decoder.decodeInstruction(slice);
|
2017-12-13 15:32:54 +01:00
|
|
|
|
2018-03-22 20:01:57 +01:00
|
|
|
analyzer.processInstruction(
|
2017-12-18 00:28:54 +01:00
|
|
|
decoder.getInstruction());
|
2017-12-13 15:32:54 +01:00
|
|
|
}
|
2018-03-22 20:01:57 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
void DxbcModule::runCompiler(
|
|
|
|
DxbcCompiler& compiler,
|
|
|
|
DxbcCodeSlice slice) const {
|
|
|
|
DxbcDecodeContext decoder;
|
2017-12-08 18:14:05 +01:00
|
|
|
|
2018-03-22 20:01:57 +01:00
|
|
|
while (!slice.atEnd()) {
|
|
|
|
decoder.decodeInstruction(slice);
|
|
|
|
|
|
|
|
compiler.processInstruction(
|
|
|
|
decoder.getInstruction());
|
|
|
|
}
|
2017-10-16 17:50:09 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
}
|