https://github.com/shader-slang/slang
Tip revision: d06a78d935b2743494d47ed5cd3f36e38ac9c5ac authored by Yong He on 04 February 2022, 03:17:30 UTC
Add gfx interop to allow more direct D3D12 usage scenarios. (#2117)
Add gfx interop to allow more direct D3D12 usage scenarios. (#2117)
Tip revision: d06a78d
slang-ir-spirv-snippet.cpp
// slang-ir-spirv-snippet.cpp
#include"slang-ir-spirv-snippet.h"
#include "../core/slang-token-reader.h"
namespace Slang
{
static SpvStorageClass translateStorageClass(String name)
{
if (name == "Uniform")
{
return SpvStorageClassUniform;
}
else if (name == "StorageBuffer")
{
return SpvStorageClassStorageBuffer;
}
return (SpvStorageClass)-1;
}
SpvSnippet::ASMType parseASMType(Slang::Misc::TokenReader& tokenReader)
{
auto word = tokenReader.ReadWord();
if (word == "float")
return SpvSnippet::ASMType::Float;
else if (word == "double")
return SpvSnippet::ASMType::Double;
else if (word == "uint2")
return SpvSnippet::ASMType::UInt2;
else if (word == "float2")
return SpvSnippet::ASMType::Float2;
else if (word == "int")
return SpvSnippet::ASMType::Int;
else if (word == "_p")
return SpvSnippet::ASMType::FloatOrDouble;
return SpvSnippet::ASMType::None;
}
RefPtr<SpvSnippet> SpvSnippet::parse(UnownedStringSlice definition)
{
RefPtr<SpvSnippet> snippet = new SpvSnippet();
try
{
Dictionary<String, SpvWord> mapInstNameToIndex;
Slang::Misc::TokenReader tokenReader(definition);
// A leading "*" at the beginning of the snip modifies $resultType with
// a storage class.
if (tokenReader.AdvanceIf("*"))
{
auto storageToken = tokenReader.ReadWord();
snippet->resultStorageClass = translateStorageClass(storageToken);
}
while (!tokenReader.IsEnd())
{
SpvSnippet::ASMInst inst;
if (tokenReader.AdvanceIf("%"))
{
String instName = tokenReader.ReadToken().Content;
mapInstNameToIndex[instName] = (int)snippet->instructions.getCount();
tokenReader.Read(Slang::Misc::TokenType::OpAssign);
}
inst.opCode = (SpvWord)tokenReader.ReadInt();
bool insideOperandList = true;
while (insideOperandList)
{
ASMOperand operand = {ASMOperandType::SpvWord, 0, 0, 0};
switch (tokenReader.NextToken().Type)
{
case Slang::Misc::TokenType::Semicolon:
insideOperandList = false;
tokenReader.ReadToken();
break;
case Slang::Misc::TokenType::IntLiteral:
operand.type = SpvSnippet::ASMOperandType::SpvWord;
operand.content = tokenReader.ReadInt();
inst.operands.add(operand);
break;
case Slang::Misc::TokenType::OpMod:
{
tokenReader.ReadToken();
operand.type = SpvSnippet::ASMOperandType::InstReference;
auto refName = tokenReader.ReadToken().Content;
if (!mapInstNameToIndex.TryGetValue(refName, operand.content))
{
SLANG_ASSERT(!"Invalid SPV ASM: referenced inst is not defined.");
}
inst.operands.add(operand);
}
break;
case Slang::Misc::TokenType::Identifier:
{
auto identifier = tokenReader.ReadToken().Content;
if (identifier == "resultType")
{
operand.type = SpvSnippet::ASMOperandType::ResultTypeId;
operand.content = (SpvWord)0xFFFFFFFF;
if (tokenReader.AdvanceIf("*"))
{
// A "*" at operand qualifies the use of `resultType` with
// a storage class, but does not modify `resultType` itself.
auto storageClass = tokenReader.ReadWord();
auto spvStorageClass = translateStorageClass(storageClass);
operand.content = spvStorageClass;
snippet->usedResultTypeStorageClasses.add(spvStorageClass);
}
inst.operands.add(operand);
}
else if (identifier == "resultId")
{
operand.type = SpvSnippet::ASMOperandType::ResultId;
inst.operands.add(operand);
}
else if (identifier == "glsl450")
{
operand.type = SpvSnippet::ASMOperandType::GLSL450ExtInstSet;
inst.operands.add(operand);
}
else if (identifier == "fi")
{
operand.type = SpvSnippet::ASMOperandType::FloatIntegerSelection;
tokenReader.Read("(");
operand.content = (SpvWord)tokenReader.ReadInt();
tokenReader.Read(",");
operand.content2 = (SpvWord)tokenReader.ReadInt();
tokenReader.Read(")");
inst.operands.add(operand);
}
else if (identifier == "fus")
{
operand.type = SpvSnippet::ASMOperandType::FloatUnsignedSignedSelection;
tokenReader.Read("(");
operand.content = (SpvWord)tokenReader.ReadInt();
tokenReader.Read(",");
operand.content2 = (SpvWord)tokenReader.ReadInt();
tokenReader.Read(",");
operand.content3 = (SpvWord)tokenReader.ReadInt();
tokenReader.Read(")");
inst.operands.add(operand);
}
else if (identifier == "_type")
{
operand.type = SpvSnippet::ASMOperandType::TypeReference;
tokenReader.Read("(");
operand.content = (SpvWord)parseASMType(tokenReader);
tokenReader.Read(")");
inst.operands.add(operand);
}
else if (identifier.startsWith("_"))
{
operand.type = SpvSnippet::ASMOperandType::ObjectReference;
operand.content = (SpvWord)StringToInt(
identifier.subString(1, identifier.getLength() - 1));
inst.operands.add(operand);
}
else if (identifier == "const")
{
operand.type = SpvSnippet::ASMOperandType::ConstantReference;
ASMConstant constant;
memset(&constant, 0, sizeof(ASMConstant));
tokenReader.Read("(");
constant.type = parseASMType(tokenReader);
int i = 0;
while (tokenReader.AdvanceIf(","))
{
switch (constant.type)
{
case ASMType::Float:
case ASMType::Float2:
case ASMType::FloatOrDouble:
constant.floatValues[i] = tokenReader.ReadFloat();
++i;
break;
default:
constant.intValues[i] = tokenReader.ReadInt();
++i;
break;
}
}
tokenReader.Read(")");
snippet->constants.add(constant);
operand.content = (SpvWord)(snippet->constants.getCount() - 1);
inst.operands.add(operand);
}
else
{
SLANG_ASSERT(!"Invalid SPV ASM operand.");
}
}
break;
default:
insideOperandList = false;
break;
}
}
snippet->instructions.add(inst);
}
}
catch (const Slang::Misc::TextFormatException&)
{
SLANG_ASSERT(!"Invalid ASM format.");
}
return snippet;
}
}