Edit

kc3-lang/angle/src/compiler/translator/OutputVulkanGLSL.cpp

Branch :

  • Show log

    Commit

  • Author : Shahbaz Youssefi
    Date : 2019-07-15 11:23:08
    Hash : b82d8633
    Message : Vulkan: Atomic counter buffer support Vulkan doesn't treat atomic counters especially, and they are emulated with atomic access to storage buffers. A single atomic counter buffer binding per pipeline is supported. All the atomic counters identify an offset within this buffer. The shader is modified to include a storage buffer definition with `uint counters[];` as the only field. A compiler pass replaces atomic counter definitions with variables that hold the corresponding offset parameter, as well as changing atomic_uint types to just uint (as the offset). Where an atomic counter variable is used, it is replaced with the offset variable (plus the array index, if array). At the same time, built-in `atomicCounter*` functions are replaced with a corresponding `atomic*` function and `memoryBarrierAtomicCounter` is replaced with `memoryBarrierBuffer`. Bug: angleproject:3566 Change-Id: Iefb3d47de6a5cb3072bfa0cb94a46ac6a886d369 Reviewed-on: https://chromium-review.googlesource.com/c/angle/angle/+/1704635 Commit-Queue: Shahbaz Youssefi <syoussefi@chromium.org> Reviewed-by: Jamie Madill <jmadill@chromium.org> Reviewed-by: Tim Van Patten <timvp@google.com>

  • src/compiler/translator/OutputVulkanGLSL.cpp
  • //
    // Copyright (c) 2016 The ANGLE Project Authors. All rights reserved.
    // Use of this source code is governed by a BSD-style license that can be
    // found in the LICENSE file.
    //
    // OutputVulkanGLSL:
    //   Code that outputs shaders that fit GL_KHR_vulkan_glsl.
    //   The shaders are then fed into glslang to spit out SPIR-V (libANGLE-side).
    //   See: https://www.khronos.org/registry/vulkan/specs/misc/GL_KHR_vulkan_glsl.txt
    //
    
    #include "compiler/translator/OutputVulkanGLSL.h"
    
    #include "compiler/translator/BaseTypes.h"
    #include "compiler/translator/Symbol.h"
    #include "compiler/translator/util.h"
    
    namespace sh
    {
    
    TOutputVulkanGLSL::TOutputVulkanGLSL(TInfoSinkBase &objSink,
                                         ShArrayIndexClampingStrategy clampingStrategy,
                                         ShHashFunction64 hashFunction,
                                         NameMap &nameMap,
                                         TSymbolTable *symbolTable,
                                         sh::GLenum shaderType,
                                         int shaderVersion,
                                         ShShaderOutput output,
                                         ShCompileOptions compileOptions)
        : TOutputGLSL(objSink,
                      clampingStrategy,
                      hashFunction,
                      nameMap,
                      symbolTable,
                      shaderType,
                      shaderVersion,
                      output,
                      compileOptions)
    {}
    
    // TODO(jmadill): This is not complete.
    void TOutputVulkanGLSL::writeLayoutQualifier(TIntermTyped *variable)
    {
        const TType &type = variable->getType();
    
        bool needsCustomLayout =
            (type.getQualifier() == EvqAttribute || type.getQualifier() == EvqFragmentOut ||
             type.getQualifier() == EvqVertexIn || IsVarying(type.getQualifier()) ||
             IsSampler(type.getBasicType()) || type.isInterfaceBlock());
    
        if (!NeedsToWriteLayoutQualifier(type) && !needsCustomLayout)
        {
            return;
        }
    
        TInfoSinkBase &out                      = objSink();
        const TLayoutQualifier &layoutQualifier = type.getLayoutQualifier();
    
        // This isn't super clean, but it gets the job done.
        // See corresponding code in GlslangWrapper.cpp.
        TIntermSymbol *symbol = variable->getAsSymbolNode();
        ASSERT(symbol);
    
        ImmutableString name      = symbol->getName();
        const char *blockStorage  = nullptr;
        const char *matrixPacking = nullptr;
    
        // For interface blocks, use the block name instead.  When the layout qualifier is being
        // replaced in the backend, that would be the name that's available.
        if (type.isInterfaceBlock())
        {
            const TInterfaceBlock *interfaceBlock = type.getInterfaceBlock();
            name                                  = interfaceBlock->name();
            TLayoutBlockStorage storage           = interfaceBlock->blockStorage();
    
            // Make sure block storage format is specified.
            if (storage != EbsStd430)
            {
                // Change interface block layout qualifiers to std140 for any layout that is not
                // explicitly set to std430.  This is to comply with GL_KHR_vulkan_glsl where shared and
                // packed are not allowed (and std140 could be used instead) and unspecified layouts can
                // assume either std140 or std430 (and we choose std140 as std430 is not yet universally
                // supported).
                storage = EbsStd140;
            }
    
            if (interfaceBlock->blockStorage() != EbsUnspecified)
            {
                blockStorage = getBlockStorageString(storage);
            }
        }
    
        // Specify matrix packing if necessary.
        if (layoutQualifier.matrixPacking != EmpUnspecified)
        {
            matrixPacking = getMatrixPackingString(layoutQualifier.matrixPacking);
        }
    
        if (needsCustomLayout)
        {
            out << "@@ LAYOUT-" << name << "(";
        }
        else
        {
            out << "layout(";
        }
    
        // Output the list of qualifiers already known at this stage, i.e. everything other than
        // `location` and `set`/`binding`.
        std::string otherQualifiers = getCommonLayoutQualifiers(variable);
    
        const char *separator = "";
        if (blockStorage)
        {
            out << separator << blockStorage;
            separator = ", ";
        }
        if (matrixPacking)
        {
            out << separator << matrixPacking;
            separator = ", ";
        }
        if (!otherQualifiers.empty())
        {
            out << separator << otherQualifiers;
        }
    
        out << ") ";
        if (needsCustomLayout)
        {
            out << "@@";
        }
    }
    
    void TOutputVulkanGLSL::writeQualifier(TQualifier qualifier,
                                           const TType &type,
                                           const TSymbol *symbol)
    {
        if (qualifier != EvqUniform && qualifier != EvqBuffer && qualifier != EvqAttribute &&
            qualifier != EvqVertexIn && !sh::IsVarying(qualifier))
        {
            TOutputGLSLBase::writeQualifier(qualifier, type, symbol);
            return;
        }
    
        if (symbol == nullptr)
        {
            return;
        }
    
        ImmutableString name = symbol->name();
    
        // For interface blocks, use the block name instead.  When the qualifier is being replaced in
        // the backend, that would be the name that's available.
        if (type.isInterfaceBlock())
        {
            name = type.getInterfaceBlock()->name();
        }
    
        TInfoSinkBase &out = objSink();
        out << "@@ QUALIFIER-" << name.data() << "(" << getMemoryQualifiers(type) << ") @@ ";
    }
    
    void TOutputVulkanGLSL::writeVariableType(const TType &type, const TSymbol *symbol)
    {
        TType overrideType(type);
    
        // External textures are treated as 2D textures in the vulkan back-end
        if (type.getBasicType() == EbtSamplerExternalOES)
        {
            overrideType.setBasicType(EbtSampler2D);
        }
    
        TOutputGLSL::writeVariableType(overrideType, symbol);
    }
    
    void TOutputVulkanGLSL::writeStructType(const TStructure *structure)
    {
        if (!structDeclared(structure))
        {
            declareStruct(structure);
            objSink() << ";\n";
        }
    }
    
    void TOutputVulkanGLSL::visitSymbol(TIntermSymbol *node)
    {
        TInfoSinkBase &out = objSink();
    
        // All the special cases are built-ins, so if it's not a built-in we can return early.
        if (node->variable().symbolType() != SymbolType::BuiltIn)
        {
            TOutputGLSL::visitSymbol(node);
            return;
        }
    
        // Some built-ins get a special translation.
        const ImmutableString &name = node->getName();
        if (name == "gl_VertexID")
        {
            // gl_VertexIndex in Vulkan GLSL has the same semantics as gl_VertexID.
            out << "gl_VertexIndex";
        }
        else if (name == "gl_InstanceID")
        {
            // gl_InstanceIndex in Vulkan GLSL is equal to
            // gl_InstanceID + baseInstance, but in OpenGL ES,
            // baseInstance is always zero.
            // (OpenGL ES 3.2 spec page 278 footnote 3)
            out << "gl_InstanceIndex";
        }
        else
        {
            TOutputGLSL::visitSymbol(node);
        }
    }
    }  // namespace sh