зеркало из https://github.com/stride3d/xkslang.git
191 строка
7.0 KiB
C++
191 строка
7.0 KiB
C++
//
|
|
// Copyright (C) 2016 LunarG, Inc.
|
|
//
|
|
// All rights reserved.
|
|
//
|
|
// Redistribution and use in source and binary forms, with or without
|
|
// modification, are permitted provided that the following conditions
|
|
// are met:
|
|
//
|
|
// Redistributions of source code must retain the above copyright
|
|
// notice, this list of conditions and the following disclaimer.
|
|
//
|
|
// Redistributions in binary form must reproduce the above
|
|
// copyright notice, this list of conditions and the following
|
|
// disclaimer in the documentation and/or other materials provided
|
|
// with the distribution.
|
|
//
|
|
// Neither the name of Google, Inc., nor the names of its
|
|
// contributors may be used to endorse or promote products derived
|
|
// from this software without specific prior written permission.
|
|
//
|
|
// THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS
|
|
// "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT
|
|
// LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS
|
|
// FOR A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE
|
|
// COPYRIGHT HOLDERS OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT,
|
|
// INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING,
|
|
// BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES;
|
|
// LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER
|
|
// CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT
|
|
// LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN
|
|
// ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE
|
|
// POSSIBILITY OF SUCH DAMAGE.
|
|
//
|
|
|
|
#include "hlslAttributes.h"
|
|
#include <cstdlib>
|
|
#include <cctype>
|
|
#include <algorithm>
|
|
|
|
namespace glslang {
|
|
// Map the given string to an attribute enum from TAttributeType,
|
|
// or EatNone if invalid.
|
|
TAttributeType TAttributeMap::attributeFromName(const TString& nameSpace, const TString& name)
|
|
{
|
|
// These are case insensitive.
|
|
TString lowername(name);
|
|
std::transform(lowername.begin(), lowername.end(), lowername.begin(), ::tolower);
|
|
TString lowernameSpace(nameSpace);
|
|
std::transform(lowernameSpace.begin(), lowernameSpace.end(), lowernameSpace.begin(), ::tolower);
|
|
|
|
// handle names within a namespace
|
|
|
|
if (lowernameSpace == "vk") {
|
|
if (lowername == "input_attachment_index")
|
|
return EatInputAttachment;
|
|
else if (lowername == "location")
|
|
return EatLocation;
|
|
else if (lowername == "binding")
|
|
return EatBinding;
|
|
else if (lowername == "global_cbuffer_binding")
|
|
return EatGlobalBinding;
|
|
else if (lowername == "builtin")
|
|
return EatBuiltIn;
|
|
else if (lowername == "constant_id")
|
|
return EatConstantId;
|
|
else if (lowername == "push_constant")
|
|
return EatPushConstant;
|
|
} else if (lowernameSpace.size() > 0)
|
|
return EatNone;
|
|
|
|
// handle names with no namespace
|
|
|
|
if (lowername == "allow_uav_condition")
|
|
return EatAllow_uav_condition;
|
|
else if (lowername == "branch")
|
|
return EatBranch;
|
|
else if (lowername == "call")
|
|
return EatCall;
|
|
else if (lowername == "domain")
|
|
return EatDomain;
|
|
else if (lowername == "earlydepthstencil")
|
|
return EatEarlyDepthStencil;
|
|
else if (lowername == "fastopt")
|
|
return EatFastOpt;
|
|
else if (lowername == "flatten")
|
|
return EatFlatten;
|
|
else if (lowername == "forcecase")
|
|
return EatForceCase;
|
|
else if (lowername == "instance")
|
|
return EatInstance;
|
|
else if (lowername == "maxtessfactor")
|
|
return EatMaxTessFactor;
|
|
else if (lowername == "maxvertexcount")
|
|
return EatMaxVertexCount;
|
|
else if (lowername == "numthreads")
|
|
return EatNumThreads;
|
|
else if (lowername == "outputcontrolpoints")
|
|
return EatOutputControlPoints;
|
|
else if (lowername == "outputtopology")
|
|
return EatOutputTopology;
|
|
else if (lowername == "partitioning")
|
|
return EatPartitioning;
|
|
else if (lowername == "patchconstantfunc")
|
|
return EatPatchConstantFunc;
|
|
else if (lowername == "unroll")
|
|
return EatUnroll;
|
|
else if (lowername == "loop")
|
|
return EatLoop;
|
|
else
|
|
return EatNone;
|
|
}
|
|
|
|
// Look up entry, inserting if it's not there, and if name is a valid attribute name
|
|
// as known by attributeFromName.
|
|
TAttributeType TAttributeMap::setAttribute(const TString& nameSpace, const TString* name, TIntermAggregate* value)
|
|
{
|
|
if (name == nullptr)
|
|
return EatNone;
|
|
|
|
const TAttributeType attr = attributeFromName(nameSpace, *name);
|
|
|
|
if (attr != EatNone)
|
|
attributes[attr] = value;
|
|
|
|
return attr;
|
|
}
|
|
|
|
// Look up entry (const version), and return aggregate node. This cannot change the map.
|
|
const TIntermAggregate* TAttributeMap::operator[](TAttributeType attr) const
|
|
{
|
|
const auto entry = attributes.find(attr);
|
|
|
|
return (entry == attributes.end()) ? nullptr : entry->second;
|
|
}
|
|
|
|
// True if entry exists in map (even if value is nullptr)
|
|
bool TAttributeMap::contains(TAttributeType attr) const
|
|
{
|
|
return attributes.find(attr) != attributes.end();
|
|
}
|
|
|
|
// extract integers out of attribute arguments stored in attribute aggregate
|
|
bool TAttributeMap::getInt(TAttributeType attr, int& value, int argNum) const
|
|
{
|
|
const TConstUnion* intConst = getConstUnion(attr, EbtInt, argNum);
|
|
|
|
if (intConst == nullptr)
|
|
return false;
|
|
|
|
value = intConst->getIConst();
|
|
return true;
|
|
};
|
|
|
|
// extract strings out of attribute arguments stored in attribute aggregate.
|
|
// convert to lower case if converToLower is true (for case-insensitive compare convenience)
|
|
bool TAttributeMap::getString(TAttributeType attr, TString& value, int argNum, bool convertToLower) const
|
|
{
|
|
const TConstUnion* stringConst = getConstUnion(attr, EbtString, argNum);
|
|
|
|
if (stringConst == nullptr)
|
|
return false;
|
|
|
|
value = *stringConst->getSConst();
|
|
|
|
// Convenience.
|
|
if (convertToLower)
|
|
std::transform(value.begin(), value.end(), value.begin(), ::tolower);
|
|
|
|
return true;
|
|
};
|
|
|
|
// Helper to get attribute const union. Returns nullptr on failure.
|
|
const TConstUnion* TAttributeMap::getConstUnion(TAttributeType attr, TBasicType basicType, int argNum) const
|
|
{
|
|
const TIntermAggregate* attrAgg = (*this)[attr];
|
|
if (attrAgg == nullptr)
|
|
return nullptr;
|
|
|
|
if (argNum >= int(attrAgg->getSequence().size()))
|
|
return nullptr;
|
|
|
|
const TConstUnion* constVal = &attrAgg->getSequence()[argNum]->getAsConstantUnion()->getConstArray()[0];
|
|
if (constVal == nullptr || constVal->getType() != basicType)
|
|
return nullptr;
|
|
|
|
return constVal;
|
|
}
|
|
|
|
} // end namespace glslang
|