mirror of
https://github.com/doitsujin/dxvk.git
synced 2024-12-12 13:08:50 +01:00
123 lines
3.4 KiB
C++
123 lines
3.4 KiB
C++
#include "dxbc_chunk_isgn.h"
|
|
|
|
namespace dxvk {
|
|
|
|
DxbcIsgn::DxbcIsgn(DxbcReader reader, DxbcTag tag) {
|
|
uint32_t elementCount = reader.readu32();
|
|
reader.skip(sizeof(uint32_t));
|
|
|
|
std::array<DxbcScalarType, 4> componentTypes = {
|
|
DxbcScalarType::Uint32, DxbcScalarType::Uint32,
|
|
DxbcScalarType::Sint32, DxbcScalarType::Float32,
|
|
};
|
|
|
|
// https://github.com/DarkStarSword/3d-fixes/blob/master/dx11shaderanalyse.py#L101
|
|
bool hasStream = (tag == "ISG1") || (tag == "OSG1") || (tag == "PSG1") || (tag == "OSG5");
|
|
bool hasPrecision = (tag == "ISG1") || (tag == "OSG1") || (tag == "PSG1");
|
|
|
|
for (uint32_t i = 0; i < elementCount; i++) {
|
|
DxbcSgnEntry entry;
|
|
entry.streamId = hasStream ? reader.readu32() : 0;
|
|
entry.semanticName = reader.clone(reader.readu32()).readString();
|
|
entry.semanticIndex = reader.readu32();
|
|
entry.systemValue = static_cast<DxbcSystemValue>(reader.readu32());
|
|
entry.componentType = componentTypes.at(reader.readu32());
|
|
entry.registerId = reader.readu32();
|
|
entry.componentMask = bit::extract(reader.readu32(), 0, 3);
|
|
|
|
if (hasPrecision)
|
|
reader.readu32();
|
|
|
|
m_entries.push_back(entry);
|
|
}
|
|
}
|
|
|
|
|
|
DxbcIsgn::~DxbcIsgn() {
|
|
|
|
}
|
|
|
|
|
|
const DxbcSgnEntry* DxbcIsgn::findByRegister(uint32_t registerId) const {
|
|
for (auto e = this->begin(); e != this->end(); e++) {
|
|
if (e->registerId == registerId)
|
|
return &(*e);
|
|
}
|
|
|
|
return nullptr;
|
|
}
|
|
|
|
|
|
const DxbcSgnEntry* DxbcIsgn::find(
|
|
const std::string& semanticName,
|
|
uint32_t semanticIndex,
|
|
uint32_t streamId) const {
|
|
for (auto e = this->begin(); e != this->end(); e++) {
|
|
if (e->semanticIndex == semanticIndex
|
|
&& e->streamId == streamId
|
|
&& compareSemanticNames(semanticName, e->semanticName))
|
|
return &(*e);
|
|
}
|
|
|
|
return nullptr;
|
|
}
|
|
|
|
|
|
DxbcRegMask DxbcIsgn::regMask(
|
|
uint32_t registerId) const {
|
|
DxbcRegMask mask;
|
|
|
|
for (auto e = this->begin(); e != this->end(); e++) {
|
|
if (e->registerId == registerId)
|
|
mask |= e->componentMask;
|
|
}
|
|
|
|
return mask;
|
|
}
|
|
|
|
|
|
uint32_t DxbcIsgn::maxRegisterCount() const {
|
|
uint32_t result = 0;
|
|
for (auto e = this->begin(); e != this->end(); e++)
|
|
result = std::max(result, e->registerId + 1);
|
|
return result;
|
|
}
|
|
|
|
void DxbcIsgn::printEntries() const {
|
|
for (auto entry = this->begin(); entry != this->end(); entry++) {
|
|
Logger::debug(str::format("SGN Entry:\n\t",
|
|
"semanticName: ", entry->semanticName, "\n\t",
|
|
"semanticIndex: ", entry->semanticIndex, "\n\t",
|
|
"registerId: ", entry->registerId, "\n\t",
|
|
"componentMask: ", entry->componentMask.maskString(), "\n\t",
|
|
"componentType: ", entry->componentType, "\n\t",
|
|
"systemValue: ", entry->systemValue, "\n\t",
|
|
"streamId: ", entry->streamId, "\n",
|
|
"\n"));
|
|
}
|
|
}
|
|
|
|
|
|
bool DxbcIsgn::compareSemanticNames(
|
|
const std::string& a, const std::string& b) {
|
|
if (a.size() != b.size())
|
|
return false;
|
|
|
|
for (size_t i = 0; i < a.size(); i++) {
|
|
char ac = a[i];
|
|
char bc = b[i];
|
|
|
|
if (ac != bc) {
|
|
if (ac >= 'A' && ac <= 'Z') ac += 'a' - 'A';
|
|
if (bc >= 'A' && bc <= 'Z') bc += 'a' - 'A';
|
|
|
|
if (ac != bc)
|
|
return false;
|
|
}
|
|
}
|
|
|
|
return true;
|
|
}
|
|
|
|
}
|