blob: f6d60a3eda5b64db8a5d647065df86889185d34e [file] [log] [blame]
/*
* Copyright 2016 Google Inc.
*
* Use of this source code is governed by a BSD-style license that can be
* found in the LICENSE file.
*/
#ifndef GrVkUniformHandler_DEFINED
#define GrVkUniformHandler_DEFINED
#include "include/gpu/vk/GrVkTypes.h"
#include "src/base/SkTBlockList.h"
#include "src/gpu/ganesh/GrSamplerState.h"
#include "src/gpu/ganesh/GrShaderVar.h"
#include "src/gpu/ganesh/glsl/GrGLSLProgramBuilder.h"
#include "src/gpu/ganesh/glsl/GrGLSLUniformHandler.h"
#include "src/gpu/ganesh/vk/GrVkSampler.h"
class GrVkUniformHandler : public GrGLSLUniformHandler {
public:
static const int kUniformsPerBlock = 8;
enum {
/**
* Binding a descriptor set invalidates all higher index descriptor sets. We must bind
* in the order of this enumeration. Samplers are after Uniforms because GrOps can specify
* GP textures as dynamic state, meaning they get rebound for each draw in a pipeline while
* uniforms are bound once before all the draws. We bind input attachments after samplers
* so those also need to be rebound if we bind new samplers.
*/
kUniformBufferDescSet = 0,
kSamplerDescSet = 1,
kInputDescSet = 2,
kLastDescSet = kInputDescSet,
};
static constexpr int kDescSetCount = kLastDescSet + 1;
// The bindings within their respective sets for various descriptor types.
enum {
kUniformBinding = 0,
kInputBinding = 0,
};
enum {
kDstInputAttachmentIndex = 0
};
// The two types of memory layout we're concerned with
enum Layout {
kStd140Layout = 0,
kStd430Layout = 1,
kLastLayout = kStd430Layout
};
static constexpr int kLayoutCount = kLastLayout + 1;
struct VkUniformInfo : public UniformInfo {
// offsets are only valid if the SkSLType of the fVariable is not a sampler.
uint32_t fOffsets[kLayoutCount];
// fImmutableSampler is used for sampling an image with a ycbcr conversion.
const GrVkSampler* fImmutableSampler = nullptr;
};
typedef SkTBlockList<VkUniformInfo> UniformInfoArray;
~GrVkUniformHandler() override;
const GrShaderVar& getUniformVariable(UniformHandle u) const override {
return fUniforms.item(u.toIndex()).fVariable;
}
const char* getUniformCStr(UniformHandle u) const override {
return this->getUniformVariable(u).c_str();
}
/**
* Returns the offset that the RTFlip synthetic uniform should use if it needs to be created.
*/
uint32_t getRTFlipOffset() const;
int numUniforms() const override {
return fUniforms.count();
}
UniformInfo& uniform(int idx) override {
return fUniforms.item(idx);
}
const UniformInfo& uniform(int idx) const override {
return fUniforms.item(idx);
}
bool usePushConstants() const { return fUsePushConstants; }
uint32_t currentOffset() const {
return fUsePushConstants ? fCurrentOffsets[kStd430Layout] : fCurrentOffsets[kStd140Layout];
}
private:
explicit GrVkUniformHandler(GrGLSLProgramBuilder* program)
: INHERITED(program)
, fUniforms(kUniformsPerBlock)
, fSamplers(kUniformsPerBlock)
, fUsePushConstants(false)
, fCurrentOffsets{0, 0} {
}
UniformHandle internalAddUniformArray(const GrProcessor* owner,
uint32_t visibility,
SkSLType type,
const char* name,
bool mangleName,
int arrayCount,
const char** outName) override;
SamplerHandle addSampler(const GrBackendFormat&,
GrSamplerState,
const skgpu::Swizzle&,
const char* name,
const GrShaderCaps*) override;
SamplerHandle addInputSampler(const skgpu::Swizzle& swizzle, const char* name) override;
int numSamplers() const { return fSamplers.count(); }
const char* samplerVariable(SamplerHandle handle) const override {
return fSamplers.item(handle.toIndex()).fVariable.c_str();
}
skgpu::Swizzle samplerSwizzle(SamplerHandle handle) const override {
return fSamplerSwizzles[handle.toIndex()];
}
uint32_t samplerVisibility(SamplerHandle handle) const {
return fSamplers.item(handle.toIndex()).fVisibility;
}
const GrVkSampler* immutableSampler(UniformHandle u) const {
return fSamplers.item(u.toIndex()).fImmutableSampler;
}
const char* inputSamplerVariable(SamplerHandle handle) const override {
// Currently we will only ever have one input sampler variable, though in the future we may
// expand to allow more inputs. For now assert that any requested handle maps to index 0,
// to make sure we didn't add multiple input samplers.
SkASSERT(handle.toIndex() == 0);
return fInputUniform.fVariable.c_str();
}
skgpu::Swizzle inputSamplerSwizzle(SamplerHandle handle) const override {
SkASSERT(handle.toIndex() == 0);
return fInputSwizzle;
}
void appendUniformDecls(GrShaderFlags, SkString*) const override;
const VkUniformInfo& getUniformInfo(UniformHandle u) const {
return fUniforms.item(u.toIndex());
}
void determineIfUsePushConstants() const;
UniformInfoArray fUniforms;
UniformInfoArray fSamplers;
skia_private::TArray<skgpu::Swizzle> fSamplerSwizzles;
UniformInfo fInputUniform;
skgpu::Swizzle fInputSwizzle;
mutable bool fUsePushConstants;
uint32_t fCurrentOffsets[kLayoutCount];
friend class GrVkPipelineStateBuilder;
friend class GrVkDescriptorSetManager;
using INHERITED = GrGLSLUniformHandler;
};
#endif