blob: a57fe4c06a9fb33a52c1aa11831b19aa3a299213 [file] [log] [blame]
/*
* Copyright 2020 Google LLC.
*
* Use of this source code is governed by a BSD-style license that can be
* found in the LICENSE file.
*/
#include "src/sksl/ir/SkSLConstructor.h"
namespace SkSL {
std::unique_ptr<Expression> Constructor::constantPropagate(const IRGenerator& irGenerator,
const DefinitionMap& definitions) {
if (this->arguments().size() == 1 && this->arguments()[0]->is<IntLiteral>()) {
const Context& context = irGenerator.fContext;
const Type& type = this->type();
int64_t intValue = this->arguments()[0]->as<IntLiteral>().value();
if (type.isFloat()) {
// promote float(1) to 1.0
return std::make_unique<FloatLiteral>(context, fOffset, intValue);
} else if (type.isInteger()) {
// promote uint(1) to 1u
return std::make_unique<IntLiteral>(fOffset, intValue, &type);
} else if (&type == context.fBool_Type.get()) {
// promote bool(k) to true/false
return std::make_unique<BoolLiteral>(context, fOffset, intValue != 0);
}
}
return nullptr;
}
bool Constructor::compareConstant(const Context& context, const Expression& other) const {
const Constructor& c = other.as<Constructor>();
const Type& myType = this->type();
const Type& otherType = c.type();
SkASSERT(myType == otherType);
if (otherType.typeKind() == Type::TypeKind::kVector) {
bool isFloat = otherType.columns() > 1 ? otherType.componentType().isFloat()
: otherType.isFloat();
for (int i = 0; i < myType.columns(); i++) {
if (isFloat) {
if (this->getFVecComponent(i) != c.getFVecComponent(i)) {
return false;
}
} else if (this->getIVecComponent(i) != c.getIVecComponent(i)) {
return false;
}
}
return true;
}
// shouldn't be possible to have a constant constructor that isn't a vector or matrix;
// a constant scalar constructor should have been collapsed down to the appropriate
// literal
SkASSERT(myType.typeKind() == Type::TypeKind::kMatrix);
for (int col = 0; col < myType.columns(); col++) {
for (int row = 0; row < myType.rows(); row++) {
if (getMatComponent(col, row) != c.getMatComponent(col, row)) {
return false;
}
}
}
return true;
}
template <typename resultType>
resultType Constructor::getVecComponent(int index) const {
SkASSERT(this->type().typeKind() == Type::TypeKind::kVector);
if (this->arguments().size() == 1 &&
this->arguments()[0]->type().typeKind() == Type::TypeKind::kScalar) {
// This constructor just wraps a scalar. Propagate out the value.
if (std::is_floating_point<resultType>::value) {
return this->arguments()[0]->getConstantFloat();
} else {
return this->arguments()[0]->getConstantInt();
}
}
// Walk through all the constructor arguments until we reach the index we're searching for.
int current = 0;
for (const std::unique_ptr<Expression>& arg : this->arguments()) {
if (current > index) {
// Somehow, we went past the argument we're looking for. Bail.
break;
}
if (arg->type().typeKind() == Type::TypeKind::kScalar) {
if (index == current) {
// We're on the proper argument, and it's a scalar; fetch it.
if (std::is_floating_point<resultType>::value) {
return arg->getConstantFloat();
} else {
return arg->getConstantInt();
}
}
current++;
continue;
}
switch (arg->kind()) {
case Kind::kConstructor: {
const Constructor& constructor = arg->as<Constructor>();
if (current + constructor.type().columns() > index) {
// We've found a constructor that overlaps the proper argument. Descend into
// it, honoring the type.
if (constructor.type().componentType().isFloat()) {
return resultType(constructor.getVecComponent<SKSL_FLOAT>(index - current));
} else {
return resultType(constructor.getVecComponent<SKSL_INT>(index - current));
}
}
break;
}
case Kind::kPrefix: {
const PrefixExpression& prefix = arg->as<PrefixExpression>();
if (current + prefix.type().columns() > index) {
// We found a prefix operator that contains the proper argument. Descend
// into it. We only support for constant propagation of the unary minus, so
// we shouldn't see any other tokens here.
SkASSERT(prefix.getOperator() == Token::Kind::TK_MINUS);
// We expect the - prefix to always be attached to a constructor.
const Constructor& constructor = prefix.operand()->as<Constructor>();
// Descend into this constructor, honoring the type.
if (constructor.type().componentType().isFloat()) {
return -resultType(constructor.getVecComponent<SKSL_FLOAT>(index -
current));
} else {
return -resultType(constructor.getVecComponent<SKSL_INT>(index - current));
}
}
break;
}
default: {
SkDEBUGFAILF("unexpected component %d { %s } in %s\n",
index, arg->description().c_str(), description().c_str());
break;
}
}
current += arg->type().columns();
}
SkDEBUGFAILF("failed to find vector component %d in %s\n", index, description().c_str());
return -1;
}
template int Constructor::getVecComponent(int) const;
template float Constructor::getVecComponent(int) const;
SKSL_FLOAT Constructor::getMatComponent(int col, int row) const {
SkDEBUGCODE(const Type& myType = this->type();)
SkASSERT(this->isCompileTimeConstant());
SkASSERT(myType.typeKind() == Type::TypeKind::kMatrix);
SkASSERT(col < myType.columns() && row < myType.rows());
if (this->arguments().size() == 1) {
const Type& argType = this->arguments()[0]->type();
if (argType.typeKind() == Type::TypeKind::kScalar) {
// single scalar argument, so matrix is of the form:
// x 0 0
// 0 x 0
// 0 0 x
// return x if col == row
return col == row ? this->arguments()[0]->getConstantFloat() : 0.0;
}
if (argType.typeKind() == Type::TypeKind::kMatrix) {
SkASSERT(this->arguments()[0]->kind() == Expression::Kind::kConstructor);
// single matrix argument. make sure we're within the argument's bounds.
if (col < argType.columns() && row < argType.rows()) {
// within bounds, defer to argument
return ((Constructor&) *this->arguments()[0]).getMatComponent(col, row);
}
// out of bounds
return 0.0;
}
}
int currentIndex = 0;
int targetIndex = col * this->type().rows() + row;
for (const auto& arg : this->arguments()) {
const Type& argType = arg->type();
SkASSERT(targetIndex >= currentIndex);
SkASSERT(argType.rows() == 1);
if (currentIndex + argType.columns() > targetIndex) {
if (argType.columns() == 1) {
return arg->getConstantFloat();
} else {
return arg->getFVecComponent(targetIndex - currentIndex);
}
}
currentIndex += argType.columns();
}
ABORT("can't happen, matrix component out of bounds");
}
int64_t Constructor::getConstantInt() const {
// We're looking for scalar integer constructors only, i.e. `int(1)`.
SkASSERT(this->arguments().size() == 1);
SkASSERT(this->type().columns() == 1);
SkASSERT(this->type().isInteger());
// The inner argument might actually be a float! `int(1.0)` is a valid cast.
const Expression& expr = *this->arguments().front();
SkASSERT(expr.type().typeKind() == Type::TypeKind::kScalar);
return expr.type().isInteger() ? expr.getConstantInt() : (int64_t)expr.getConstantFloat();
}
SKSL_FLOAT Constructor::getConstantFloat() const {
// We're looking for scalar integer constructors only, i.e. `float(1.0)`.
SkASSERT(this->arguments().size() == 1);
SkASSERT(this->type().columns() == 1);
SkASSERT(this->type().isFloat());
// The inner argument might actually be an integer! `float(1)` is a valid cast.
const Expression& expr = *this->arguments().front();
SkASSERT(expr.type().typeKind() == Type::TypeKind::kScalar);
return expr.type().isFloat() ? expr.getConstantFloat() : (SKSL_FLOAT)expr.getConstantInt();
}
} // namespace SkSL