mirror of
https://github.com/ZDoom/ZDRay.git
synced 2025-02-18 09:21:28 +00:00
257 lines
8.2 KiB
C++
257 lines
8.2 KiB
C++
//
|
|
// Copyright (C) 2017 LunarG, Inc.
|
|
// Copyright (C) 2018 Google, 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 "attribute.h"
|
|
#include "../Include/intermediate.h"
|
|
#include "ParseHelper.h"
|
|
|
|
namespace glslang {
|
|
|
|
// extract integers out of attribute arguments stored in attribute aggregate
|
|
bool TAttributeArgs::getInt(int& value, int argNum) const
|
|
{
|
|
const TConstUnion* intConst = getConstUnion(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 TAttributeArgs::getString(TString& value, int argNum, bool convertToLower) const
|
|
{
|
|
const TConstUnion* stringConst = getConstUnion(EbtString, argNum);
|
|
|
|
if (stringConst == nullptr)
|
|
return false;
|
|
|
|
value = *stringConst->getSConst();
|
|
|
|
// Convenience.
|
|
if (convertToLower)
|
|
std::transform(value.begin(), value.end(), value.begin(), ::tolower);
|
|
|
|
return true;
|
|
}
|
|
|
|
// How many arguments were supplied?
|
|
int TAttributeArgs::size() const
|
|
{
|
|
return args == nullptr ? 0 : (int)args->getSequence().size();
|
|
}
|
|
|
|
// Helper to get attribute const union. Returns nullptr on failure.
|
|
const TConstUnion* TAttributeArgs::getConstUnion(TBasicType basicType, int argNum) const
|
|
{
|
|
if (args == nullptr)
|
|
return nullptr;
|
|
|
|
if (argNum >= (int)args->getSequence().size())
|
|
return nullptr;
|
|
|
|
const TConstUnion* constVal = &args->getSequence()[argNum]->getAsConstantUnion()->getConstArray()[0];
|
|
if (constVal == nullptr || constVal->getType() != basicType)
|
|
return nullptr;
|
|
|
|
return constVal;
|
|
}
|
|
|
|
// Implementation of TParseContext parts of attributes
|
|
TAttributeType TParseContext::attributeFromName(const TString& name) const
|
|
{
|
|
if (name == "branch" || name == "dont_flatten")
|
|
return EatBranch;
|
|
else if (name == "flatten")
|
|
return EatFlatten;
|
|
else if (name == "unroll")
|
|
return EatUnroll;
|
|
else if (name == "loop" || name == "dont_unroll")
|
|
return EatLoop;
|
|
else if (name == "dependency_infinite")
|
|
return EatDependencyInfinite;
|
|
else if (name == "dependency_length")
|
|
return EatDependencyLength;
|
|
else
|
|
return EatNone;
|
|
}
|
|
|
|
// Make an initial leaf for the grammar from a no-argument attribute
|
|
TAttributes* TParseContext::makeAttributes(const TString& identifier) const
|
|
{
|
|
TAttributes *attributes = nullptr;
|
|
attributes = NewPoolObject(attributes);
|
|
TAttributeArgs args = { attributeFromName(identifier), nullptr };
|
|
attributes->push_back(args);
|
|
return attributes;
|
|
}
|
|
|
|
// Make an initial leaf for the grammar from a one-argument attribute
|
|
TAttributes* TParseContext::makeAttributes(const TString& identifier, TIntermNode* node) const
|
|
{
|
|
TAttributes *attributes = nullptr;
|
|
attributes = NewPoolObject(attributes);
|
|
|
|
// for now, node is always a simple single expression, but other code expects
|
|
// a list, so make it so
|
|
TIntermAggregate* agg = intermediate.makeAggregate(node);
|
|
TAttributeArgs args = { attributeFromName(identifier), agg };
|
|
attributes->push_back(args);
|
|
return attributes;
|
|
}
|
|
|
|
// Merge two sets of attributes into a single set.
|
|
// The second argument is destructively consumed.
|
|
TAttributes* TParseContext::mergeAttributes(TAttributes* attr1, TAttributes* attr2) const
|
|
{
|
|
attr1->splice(attr1->end(), *attr2);
|
|
return attr1;
|
|
}
|
|
|
|
//
|
|
// Selection attributes
|
|
//
|
|
void TParseContext::handleSelectionAttributes(const TAttributes& attributes, TIntermNode* node)
|
|
{
|
|
TIntermSelection* selection = node->getAsSelectionNode();
|
|
if (selection == nullptr)
|
|
return;
|
|
|
|
for (auto it = attributes.begin(); it != attributes.end(); ++it) {
|
|
if (it->size() > 0) {
|
|
warn(node->getLoc(), "attribute with arguments not recognized, skipping", "", "");
|
|
continue;
|
|
}
|
|
|
|
switch (it->name) {
|
|
case EatFlatten:
|
|
selection->setFlatten();
|
|
break;
|
|
case EatBranch:
|
|
selection->setDontFlatten();
|
|
break;
|
|
default:
|
|
warn(node->getLoc(), "attribute does not apply to a selection", "", "");
|
|
break;
|
|
}
|
|
}
|
|
}
|
|
|
|
//
|
|
// Switch attributes
|
|
//
|
|
void TParseContext::handleSwitchAttributes(const TAttributes& attributes, TIntermNode* node)
|
|
{
|
|
TIntermSwitch* selection = node->getAsSwitchNode();
|
|
if (selection == nullptr)
|
|
return;
|
|
|
|
for (auto it = attributes.begin(); it != attributes.end(); ++it) {
|
|
if (it->size() > 0) {
|
|
warn(node->getLoc(), "attribute with arguments not recognized, skipping", "", "");
|
|
continue;
|
|
}
|
|
|
|
switch (it->name) {
|
|
case EatFlatten:
|
|
selection->setFlatten();
|
|
break;
|
|
case EatBranch:
|
|
selection->setDontFlatten();
|
|
break;
|
|
default:
|
|
warn(node->getLoc(), "attribute does not apply to a switch", "", "");
|
|
break;
|
|
}
|
|
}
|
|
}
|
|
|
|
//
|
|
// Loop attributes
|
|
//
|
|
void TParseContext::handleLoopAttributes(const TAttributes& attributes, TIntermNode* node)
|
|
{
|
|
TIntermLoop* loop = node->getAsLoopNode();
|
|
if (loop == nullptr) {
|
|
// the actual loop might be part of a sequence
|
|
TIntermAggregate* agg = node->getAsAggregate();
|
|
if (agg == nullptr)
|
|
return;
|
|
for (auto it = agg->getSequence().begin(); it != agg->getSequence().end(); ++it) {
|
|
loop = (*it)->getAsLoopNode();
|
|
if (loop != nullptr)
|
|
break;
|
|
}
|
|
if (loop == nullptr)
|
|
return;
|
|
}
|
|
|
|
for (auto it = attributes.begin(); it != attributes.end(); ++it) {
|
|
if (it->name != EatDependencyLength && it->size() > 0) {
|
|
warn(node->getLoc(), "attribute with arguments not recognized, skipping", "", "");
|
|
continue;
|
|
}
|
|
|
|
int value;
|
|
switch (it->name) {
|
|
case EatUnroll:
|
|
loop->setUnroll();
|
|
break;
|
|
case EatLoop:
|
|
loop->setDontUnroll();
|
|
break;
|
|
case EatDependencyInfinite:
|
|
loop->setLoopDependency(TIntermLoop::dependencyInfinite);
|
|
break;
|
|
case EatDependencyLength:
|
|
if (it->size() == 1 && it->getInt(value)) {
|
|
if (value <= 0)
|
|
error(node->getLoc(), "must be positive", "dependency_length", "");
|
|
loop->setLoopDependency(value);
|
|
} else
|
|
warn(node->getLoc(), "expected a single integer argument", "dependency_length", "");
|
|
break;
|
|
default:
|
|
warn(node->getLoc(), "attribute does not apply to a loop", "", "");
|
|
break;
|
|
}
|
|
}
|
|
}
|
|
|
|
|
|
} // end namespace glslang
|