Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

Add support to convert DXR HLSL to SPV_NV_ray_tracing #1920

Merged
merged 18 commits into from
Feb 27, 2019

Conversation

alelenv
Copy link
Contributor

@alelenv alelenv commented Feb 11, 2019

@ehsannas @antiagainst
Ehsan, Lei, please review the changes

Pattern match tests have been added for new features

Let us know if anything more is required from us
Thanks!

@msftclas
Copy link

msftclas commented Feb 11, 2019

CLA assistant check
All CLA requirements met.

@AppVeyorBot
Copy link

@ehsannas ehsannas self-requested a review February 12, 2019 15:52
@ehsannas ehsannas added the spirv Work related to SPIR-V label Feb 12, 2019
@dneto0
Copy link
Collaborator

dneto0 commented Feb 12, 2019

@alelenv : Thanks for this contribution!

For better inclusiveness, I'd prefer to see a copy of the HLSL functional spec in a more plain-text format, for better inclusiveness. Yes the .docx renders apparently correctly on my Mac, but it seems to be straining to do so.

I'm also a little concerned that the reference is on someone's personal .org page, and could disappear at a whim. Ideally a copy would be checked in with this PR (in the docs subtree). However, that doc lacks a copyright or author notice, so ... check with the author?

I don't mean to come off as overly nit-picky, but I'd like this to be a stable documented feature of the compiler.

@alelenv alelenv force-pushed the NV_ray_tracing_final branch from 5c5b9a7 to 549193c Compare February 12, 2019 18:15
@alelenv alelenv force-pushed the NV_ray_tracing_final branch from b9bb79b to b5dcf5c Compare February 12, 2019 19:20
@AppVeyorBot
Copy link

Copy link
Contributor

@ehsannas ehsannas left a comment

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Hi @alelenv
Thanks for your contribution. It's great to see this work. The code looks good overall, but needs some improvements.

You'll find that many of my comments are related to adhering to the C++ style guide.

  • The style guide requires proper Punctuation, Spelling and Grammar. For example, comments should be proper sentences ending with a period. Consistently use SPIR-V, rather than spirv or spir-v or Spir-v, etc in the comments. I know this sounds nit-picky, but it'll lead to a much nicer code base over time (as it has happened over the development period of Spiregg).

  • clang-format should be applied to all the files that you have touched. (you can use this extension if you're using VisualStudio)

  • I'm very glad to see you have added new tests \o/ These will prevent regressions in RT code generation in the future. Please add more if you can.
    Here are a few that I thought might be useful:
    What happens if more than 1 function has the same [shader(X)] attribute? Add a test for it.
    Add a test that exercises the matrix transpose logic in processRayBuiltins.
    etc.
    Taking testing one step further: Would you be able to test the resulting SPIR-V on hardware to ensure correctness? I don't have access to the GPU that has the RT features. That would be the ultimate test for your work.

  • SPIR-V.rst needs to be updated (we'll discuss this more on the next round of code-review).

  • Please do not force push to your branch when addressing code-review comments. Please make new commits, so that we can review the diff more easily.

  • I'll do (at least one) more rounds of code-review and testing before landing.

Thanks!

@@ -268,6 +269,11 @@ bool isOrContainsNonFpColMajorMatrix(const ASTContext &,
/// matrix type.
QualType getComponentVectorType(const ASTContext &, QualType matrixType);

/// \brief Return a QualType corresponding to HLSL matrix of given element type
Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

s/Return/Returns

@@ -420,6 +420,12 @@ class SpirvBuilder {

void createLineInfo(SpirvString *file, uint32_t line, uint32_t column);

/// \brief Create spirv instructions for NV raytracing ops
Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Use

/// \brief Creates SPIR-V instructions for NV raytracing ops.

@@ -1757,6 +1758,29 @@ class SpirvLineInfo : public SpirvInstruction {
uint32_t column;
};


/// \brief Base class for all NV raytracing instructions
/// These include following spirv opcodes
Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

/// These include the following SPIR-V opcodes:

@@ -1012,5 +1016,75 @@ QualType getComponentVectorType(const ASTContext &astContext,
return astContext.getExtVectorType(elemType, colCount);
}

QualType getHLSLMatrixType(ASTContext& astContext, Sema &S,
Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

This is very useful. Thanks for creating the custom matrix QualType 👍

QualType getHLSLMatrixType(ASTContext& astContext, Sema &S,
ClassTemplateDecl *templateDecl,
QualType elemType, int rows, int columns)
{
Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

the { should be on the previous line.
It looks like you haven't run clang-format ? If not, please do. Thanks.

ExecutionModel(ShaderKind sk, ExecModel em) : shaderKind(sk), execModel(em) {}

static const unsigned numExecutionModels = 14;
static const ExecutionModel executionModels[numExecutionModels];
Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

In general static member variables are forbidden due to their lifetime and static initialization issues.

static functions are OK.

So, there are a couple of different ways you can address this:

  1. remove these static data members. Have the static getter functions construct an ExecutionModel object and return it. e.g.
static ExecutionModel GetByStageName(llvm::StringRef stageName);
static ExecutionModel GetByShaderKind(ShaderKind sk);

there'll be switch statements inside these functions I'd presume.

  1. Another way -- if we really really want to have only 14 objects and reuse them -- would be to use SpirvContext. The context class keeps all the objects that live for the lifetime of the backend. Given how lightweight the objects of this class are, I don't think it's necessary to go down this route.

Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Can this class be fully replaced with a standalone function (in SpirvEmitter.cpp) that returns the SPIR-V execution model for a given shader model kind?
e.g.

spv::ExecutionModel getSpirvExecutionModel(hlsl::ShaderModel::Kind smk);

bool IsRay() const {
return execModel >= ExecModel::RayGenerationNV &&
execModel <= ExecModel::CallableNV;
}
Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

All these functions are reimplementing the logic in DxilShaderModel.h.
I wonder if we could just use the hlsl::ShaderModel instead of the ExecutionModel throughout the spiregg code, and remove ExecutionModel class completely? Is there a good reason for this class to exist?

Copy link
Contributor

@sparmarNV sparmarNV Feb 19, 2019

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

We can't use hlsl::ShaderModel because there are no ShaderModel objects for RT profiles in DxilShaderModel.cpp (rgen_6_4, miss_6_4, etc.).
So I implemented a similar class (ExecutionModel) which would allow me to track individual RT shader stages.
Nonetheless, I have removed ExecutionModel class and used ShaderModel::Kind to track current entry point.
Let me know if this change works for you.

@@ -380,6 +381,17 @@ class FunctionType : public SpirvType {
llvm::SmallVector<const SpirvType *, 8> paramTypes;
};

/// Represents accleration structure type as defined in SPV_NV_ray_tracing
Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

SPV_NV_ray_tracing.

SpirvFunction *entryFunction;
bool isEntryFunction;

FunctionInfo()
Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

you can remove this constructor.

if (const auto *shaderAttr = funcDecl->getAttr<HLSLShaderAttr>()) {
// If we are compiling as a library then add everything that has a
// ShaderAttr
FunctionInfo *entryInfo = new FunctionInfo(
Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

As you may have noticed, we don't perform new memory allocation anywhere except in SpirvContext (which uses placement new) which is responsible for managing all allocated memory and releasing them.

I suggest you change your data structure from

llvm::DenseMap<const DeclaratorDecl *, const FunctionInfo *> functionInfoMap;

to

llvm::DenseMap<const DeclaratorDecl *, FunctionInfo> functionInfoMap;

and then

functionInfoMap[funcDecl] = FunctionInfo(...);
workQueue.insert(&functionInfoMap[funcDecl]);

... hmmm... it looks like you can further simplify this:

Remove functionInfoMap and change

llvm::SetVector<const DeclaratorDecl *> workQueue;

to

llvm::MapVector<const DeclaratorDecl *, FunctionInfo> workQueue;

this way, workQueue[funcDecl] is the same as what funcMapInfo[funcDecl] used to be ; and if I understand llvm::MapVector correctly, you can iterate over it the same as as SetVector, so you can still have the loop over workQueue.

Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

@sparmarNV
I'm doing further testing on this PR, and I'm running into some failures. They are due to this suggestion that I made:

functionInfoMap[funcDecl] = FunctionInfo(...);
workQueue.insert(&functionInfoMap[funcDecl]);

It turns out llvm::DenseMap may choose to overwrite that address (&functionInfoMap[x]) when handling its buckets. Sorry about that. I have made the fix, and I'll post my fix to this PR directly. I have created a wrapper function that updates functionInfoMap and workQueue at the same time, and it uses placement new to allocate memory from the SpirvContext memory pool.

Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Thanks for testing our changes and I am glad that you were able to triage and revert this change before the merge.

@alelenv
Copy link
Contributor Author

alelenv commented Feb 13, 2019

@dneto0 : Sure. We are working on providing you a text copy soon which we can check into docs.

@ehsannas : Thanks for such a detailed review.
Some quick answers below. sparmarNV will be handling issues related to multiple entry points support.
We will update branch based on your feedback soon.

Q) Add a test that exercises the matrix transpose logic in processRayBuiltins. etc.
ObjectToWorld3x4() & WorldToObject3x4() are transposes of SPIR-V Builtin
A) This is already tested in raytracing.nv.closesthit.hlsl/raytracing.nv.anyhit.hlsl

Q) Taking testing one step further: Would you be able to test the resulting SPIR-V on hardware to ensure correctness? I don't have access to the GPU that has the RT features. That would be the ultimate test for your work.
A) We have some internal apps running on HW using the generated SPIR-V.

@AppVeyorBot
Copy link

@ehsannas
Copy link
Contributor

A) This is already tested in raytracing.nv.closesthit.hlsl/raytracing.nv.anyhit.hlsl

Cool! I couldn't find OpTranspose in your CHECK: commands in the tests. So it would be good to add a CHECK. This will firstly ensure OpTranspose is actually called (and not accidentally removed in the future), and it'll also be good to have a check to see the custom matrix type (e.g. 4x3 or 3x4) as the result type in a CHECK.

A) We have some internal apps running on HW using the generated SPIR-V.

Awesome! Glad to hear that.

Update tests to add multple entry functions of same shader stage
@alelenv
Copy link
Contributor Author

alelenv commented Feb 19, 2019

HLSL Spec is provided by Microsoft to users registered to DX forums
Please follow instructions provided below to register
http://forums.directxtech.com/index.php?topic=5985.0

I have updated link above to publicly available HLSL specs from MSDN

@AppVeyorBot
Copy link

- This change removes ExecutionModel class and relies on ShaderModel::Kind to track current entry point shader stage
- Also instead of declaring it in SpirvEmitter, DeclResultIdMapper & GlPerVertex, we declare it only once in common object SpirvContext
@AppVeyorBot
Copy link

Copy link
Contributor

@ehsannas ehsannas left a comment

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

@sparmarNV

Thanks for making this change 👍 . This is much nicer than what was there before.
You have eliminated a whole class, eliminated the use of static members, eliminated copies of ExecutionModel in different places and concentrated it all in SpirvContext. All great changes 👍 .

Please run clang-format if you haven't already.

declIdMapper.setSpvExecutionModel(curEntryOrCallee->spvExecModel);
spvExecModel = curEntryOrCallee->spvExecModel;
entryFunction = curEntryOrCallee->entryFunction;
spvContext.setCurrentShaderModelKind(curEntryOrCallee->shaderModelKind);
Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Great to see the 3 lines is replaced with 1 line. Shader kind is now only stored in 1 place (SpirvContext), and this makes the code less error-prone. 👍

// Current ShaderModelKind for entry point.
ShaderModelKind curShaderModelKind;
// Major/Minor hlsl profile version.
unsigned majorVersion;
Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

use uint32_t instead of unsigned to be consistent with the rest of the code.

Copy link
Contributor Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

@sparmarNV : Could you comment on this?

Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

I see both unsigned and uint32_t being used in the file.
Nonetheless, I will update the type to uint32_t.

Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Looks like a few unsigned have snuck through in the past. But let's strive to stay consistent. I'll update the other ones to also use uint32_t in a different PR. Thanks for updating.

@@ -366,12 +362,6 @@ class DeclResultIdMapper {
/// \brief Sets the entry function.
void setEntryFunction(SpirvFunction *fn) { entryFunction = fn; }

/// \brief Sets the SPIR-V execution model
Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Happy to see this is gone.

@@ -648,15 +638,13 @@ class DeclResultIdMapper {
inline bool isInputStorageClass(const StageVar &v);

private:
const hlsl::ShaderModel &shaderModel;
Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Happy to see this is gone.

SpirvBuilder &spvBuilder;
SpirvEmitter &theEmitter;
const SpirvCodeGenOptions &spirvOptions;
ASTContext &astContext;
SpirvContext &spvContext;
DiagnosticsEngine &diags;
SpirvFunction *entryFunction;
const ExecutionModel *spvExecModel;
Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Happy to see this is gone.

ExecutionModel::GetByShaderKind(shaderModel.GetKind()), funcDecl,
/*entryFunction*/ nullptr, /*isEntryFunc*/ true);
FunctionInfo *entryInfo =
new FunctionInfo(spvContext.getCurrentShaderModelKind(), funcDecl,
Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

I believe new should be avoided here. See my comment about workQueue

Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Yes, I am working on this next. Thanks for the review 👍

Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Cool. Thanks 👍

Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

I tried replacing functionInfoMap and workQueue with a MapVector, but it seems MapVector has few limitations.
We can't access the vector elements inside the MapVector with an index, which doesn't work in below case -

// The queue can grow in the meanwhile; so need to keep evaluating
// workQueue.size().
for (uint32_t i = 0; i < workQueue.size(); ++i) {
const FunctionInfo *curEntryOrCallee = workQueue[i];
...
}

Also using iterator on MapVector doesn't work with growing queue.
So I have gone with your option 1, which is to remove 'new' memory allocation by changing the DenseMap decl to -

llvm::DenseMap<const DeclaratorDecl *, FunctionInfo> functionInfoMap;

Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

@sparmarNV Ah you're right. It is not possible to iterate over a MapVector when it is growing. Thanks for removing the usage of new.

@@ -613,14 +618,19 @@ void SpirvEmitter::HandleTranslationUnit(ASTContext &context) {
spvBuilder.setMemoryModel(spv::AddressingModel::Logical,
spv::MemoryModel::GLSL450);

// Even though the 'workQueue' grows due to the above loop, the first
// 'numEntryPoints' entries in the 'workQueue' are the ones with the HLSL
// 'shader' attribute, and must therefore be entry functions.
Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Thanks for adding.

FunctionInfo *calleeInfo = new FunctionInfo(
spvExecModel, callee, /*entryFunction*/ nullptr, /*isEntryFunc*/ false);
FunctionInfo *calleeInfo =
new FunctionInfo(spvContext.getCurrentShaderModelKind(), callee,
Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

I believe new should be avoided here. See my comment about workQueue

@@ -593,6 +592,11 @@ class SpirvEmitter : public ASTConsumer {
/// Emits an error if the given attribute is not a loop attribute.
spv::LoopControlMask translateLoopAttribute(const Stmt *, const Attr &);

static hlsl::ShaderModel::Kind
SpirvEmitter::getShaderModelKind(StringRef stageName);
Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

No need for SpirvEmitter::

static hlsl::ShaderModel::Kind getShaderModelKind(StringRef stageName);

This is caught by buildbots on Linux and macOS
Same for getSpirvShaderStage below.

@@ -1,62 +0,0 @@
//===------- ExecutionModel.h - get/set SPV Execution Model -----*- C++ -*-===//
Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Happy to see that the class is gone. Especially the static data that was included in it.

@AppVeyorBot
Copy link

This change also -
- removes invalid "SpirvEmitter::" from function declarations in SpirvEmitter class.
- fix build errors by adding a default constructor in FunctionInfo struct to allow functionInfoMap allocate an empty object for no search results.
@AppVeyorBot
Copy link

@alelenv
Copy link
Contributor Author

alelenv commented Feb 22, 2019

@ehsannas : I believe we have completed updates based on your feedback for first round of reviews
Can you point us to next steps i.e how to update SPIR-V.rst?
Thanks

Copy link
Contributor

@ehsannas ehsannas left a comment

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

@alelenv Thanks for addressing my comments so far. If you go through all the files, there are a few (mostly formatting) ones left to fix.

We're almost there.

Regarding documentation:
We have a comprehensive doc file about mapping HLSL concepts to SPIR-V. The file (SPIR-V.rst) is located here. You should update this file to reflect the changes you have made in this PR.

I think it'd be good to add a top-level section about Ray Tracing between Shader Model 6.0 Wave Intrinsics and Supported Command-line Options sections. And in that Ray Tracing section, feel free to add as many sub-sections as you need. The goal is to highlight the the RTX concepts/features and how they map from HLSL to SPIR-V. In other sections of this document we've tried to provide simple explanations for people to get started, and provide external links for full details. For example, you can also provide a link to the RTX HLSL spec, and the SPIR-V NV RT extension spec. It'd also be really nice to provide a link to sample shaders (e.g. raytracing helloworld shaders) that people can use to get started.

Under HLSL Shader Stages, you should add a Ray Tracing Stages subsection, and add the newly supported shader stages under that. For each stage, it'd be good to provide some explanation of the stage and also list the expected shader inputs and outputs (stage variables). Including Figure 1 and Figure 5 from @nsubtil's blog post could be very useful here.

In general I found that blog post to be quite a good introduction. You could use some of its content as brief explanations for the new documentation you are adding to SPIR-V.rst.

Thanks!

@@ -34,7 +34,8 @@ bool SpirvFunction::invokeVisitor(Visitor *visitor, bool reverseOrder) {
if (!basicBlocks.empty()) {
BlockReadableOrderVisitor([&orderedBlocks](SpirvBasicBlock *block) {
orderedBlocks.push_back(block);
}).visit(basicBlocks.front());
})
Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

revert to previous format (clang-format).

@@ -1046,6 +1084,16 @@ class SpirvEmitter : public ASTConsumer {
/// Maps a given statement to the basic block that is associated with it.
llvm::DenseMap<const Stmt *, SpirvBasicBlock *> stmtBasicBlock;

/// Maintains mapping from a type to spirv variable along with spirv
Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

s/spirv/SPIR-V

@@ -420,6 +420,12 @@ class SpirvBuilder {

void createLineInfo(SpirvString *file, uint32_t line, uint32_t column);

/// \brief Creates spirv instructions for NV raytracing ops
Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

s/spirv/SPIR-V.
ops.

SpirvInstruction *retVal =
declIdMapper.getBuiltinVar(builtin, builtinType, callExpr->getExprLoc());
retVal = spvBuilder.createLoad(builtinType, retVal);
if (transposeMatrix)
Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

This is done now. Thanks.

spvBuilder.createLoad(hitAttributeArg->getType(), hitAttributeArgInst);
spvBuilder.createStore(hitAttributeStageVar, tempLoad);

// SPV Instruction :
Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

s/SPV/SPIR-V

builtin = spv::BuiltIn::WorldToObjectNV;
break;
default:
emitError("ray intrinsic function unimplemented", callExpr->getExprLoc());
Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Should we return nullptr after emitError?

spvBuilder.createLoad(callDataArg->getType(), callDataArgInst);
spvBuilder.createStore(callDataStageVar, tempLoad);

// SPV Instruction
Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

s/SPV/SPIR-V

@AppVeyorBot
Copy link

@AppVeyorBot
Copy link

@AppVeyorBot
Copy link

@ehsannas
Copy link
Contributor

@alelenv @sparmarNV Thanks for your updates.
Sorry I forgot to mention one thing about the doc file:
There's a Intrinsic functions section in the doc file. Please add a subsection for Ray Tracing Intrinsics, and put in a table (example) to show the HLSL-->SPIR-V mapping for the new intrinsics that you have added support for.
Thanks!

@AppVeyorBot
Copy link

@alelenv
Copy link
Contributor Author

alelenv commented Feb 25, 2019

@ehsannas : Sorry for missing a large number of formatting fixes from previous comments. My tracking changes for each comment via emails didnt work out properly.
I have updated rst with raytracing info, mappings and flowchart diagrams
Let me know your thoughts

Thanks!

@AppVeyorBot
Copy link

@ehsannas
Copy link
Contributor

Just finished reading the documentation. Awesome job, @alelenv 👍. This is exactly what we were looking for.
It provides a good overview and contains links for readers to go off and learn more (specs and tutorials).

Copy link
Contributor

@ehsannas ehsannas left a comment

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

LGTM. Thanks for addressing all the code review comments \o/
I will do a bit of more testing, and if all goes well, I'll merge the PR (hopefully by tomorrow).

@AppVeyorBot
Copy link

@AppVeyorBot
Copy link

Sign up for free to join this conversation on GitHub. Already have an account? Sign in to comment
Labels
spirv Work related to SPIR-V
Projects
None yet
Development

Successfully merging this pull request may close these issues.

6 participants