blob: d1c3d8fd62cc481d552e30e40587ac003f7f776b [file] [log] [blame]
/*
* Copyright 2021 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/SkSLTernaryExpression.h"
#include "src/sksl/SkSLAnalysis.h"
#include "src/sksl/SkSLBuiltinTypes.h"
#include "src/sksl/SkSLConstantFolder.h"
#include "src/sksl/SkSLContext.h"
#include "src/sksl/SkSLErrorReporter.h"
#include "src/sksl/SkSLOperator.h"
#include "src/sksl/SkSLProgramSettings.h"
#include "src/sksl/ir/SkSLBinaryExpression.h"
#include "src/sksl/ir/SkSLConstructorScalarCast.h"
#include "src/sksl/ir/SkSLLiteral.h"
#include "src/sksl/ir/SkSLPrefixExpression.h"
namespace SkSL {
std::unique_ptr<Expression> TernaryExpression::Convert(const Context& context,
Position pos,
std::unique_ptr<Expression> test,
std::unique_ptr<Expression> ifTrue,
std::unique_ptr<Expression> ifFalse) {
test = context.fTypes.fBool->coerceExpression(std::move(test), context);
if (!test || !ifTrue || !ifFalse) {
return nullptr;
}
if (ifTrue->type().componentType().isOpaque()) {
context.fErrors->error(pos, "ternary expression of opaque type '" +
ifTrue->type().displayName() + "' is not allowed");
return nullptr;
}
const Type* trueType;
const Type* falseType;
const Type* resultType;
Operator equalityOp(Operator::Kind::EQEQ);
if (!equalityOp.determineBinaryType(context, ifTrue->type(), ifFalse->type(),
&trueType, &falseType, &resultType) ||
!trueType->matches(*falseType)) {
Position errorPos = ifTrue->fPosition.rangeThrough(ifFalse->fPosition);
if (ifTrue->type().isVoid()) {
context.fErrors->error(errorPos, "ternary expression of type 'void' is not allowed");
} else {
context.fErrors->error(errorPos, "ternary operator result mismatch: '" +
ifTrue->type().displayName() + "', '" +
ifFalse->type().displayName() + "'");
}
return nullptr;
}
if (trueType->isOrContainsArray()) {
context.fErrors->error(pos, "ternary operator result may not be an array (or struct "
"containing an array)");
return nullptr;
}
ifTrue = trueType->coerceExpression(std::move(ifTrue), context);
if (!ifTrue) {
return nullptr;
}
ifFalse = falseType->coerceExpression(std::move(ifFalse), context);
if (!ifFalse) {
return nullptr;
}
return TernaryExpression::Make(context, pos, std::move(test), std::move(ifTrue),
std::move(ifFalse));
}
std::unique_ptr<Expression> TernaryExpression::Make(const Context& context,
Position pos,
std::unique_ptr<Expression> test,
std::unique_ptr<Expression> ifTrue,
std::unique_ptr<Expression> ifFalse) {
SkASSERT(ifTrue->type().matches(ifFalse->type()));
SkASSERT(!ifTrue->type().componentType().isOpaque());
SkASSERT(!context.fConfig->strictES2Mode() || !ifTrue->type().isOrContainsArray());
const Expression* testExpr = ConstantFolder::GetConstantValueForVariable(*test);
if (testExpr->isBoolLiteral()) {
// static boolean test, just return one of the branches
if (testExpr->as<Literal>().boolValue()) {
ifTrue->fPosition = pos;
return ifTrue;
} else {
ifFalse->fPosition = pos;
return ifFalse;
}
}
if (context.fConfig->fSettings.fOptimize) {
const Expression* ifTrueExpr = ConstantFolder::GetConstantValueForVariable(*ifTrue);
const Expression* ifFalseExpr = ConstantFolder::GetConstantValueForVariable(*ifFalse);
// A ternary with matching true- and false-cases does not need to branch.
if (Analysis::IsSameExpressionTree(*ifTrueExpr, *ifFalseExpr)) {
// If `test` has no side-effects, we can eliminate it too, and just return `ifTrue`.
if (!Analysis::HasSideEffects(*test)) {
ifTrue->fPosition = pos;
return ifTrue;
}
// Return a comma-expression containing `(test, ifTrue)`.
return BinaryExpression::Make(context, pos, std::move(test),
Operator::Kind::COMMA, std::move(ifTrue));
}
// A ternary of the form `test ? expr : false` can be simplified to `test && expr`.
if (ifFalseExpr->isBoolLiteral() && !ifFalseExpr->as<Literal>().boolValue()) {
return BinaryExpression::Make(context, pos, std::move(test),
Operator::Kind::LOGICALAND, std::move(ifTrue));
}
// A ternary of the form `test ? true : expr` can be simplified to `test || expr`.
if (ifTrueExpr->isBoolLiteral() && ifTrueExpr->as<Literal>().boolValue()) {
return BinaryExpression::Make(context, pos, std::move(test),
Operator::Kind::LOGICALOR, std::move(ifFalse));
}
// A ternary of the form `test ? false : true` can be simplified to `!test`.
if (ifTrueExpr->isBoolLiteral() && !ifTrueExpr->as<Literal>().boolValue() &&
ifFalseExpr->isBoolLiteral() && ifFalseExpr->as<Literal>().boolValue()) {
return PrefixExpression::Make(context, pos, Operator::Kind::LOGICALNOT,
std::move(test));
}
// A ternary of the form `test ? 1 : 0` can be simplified to `cast(test)`.
if (ifTrueExpr->is<Literal>() && ifTrueExpr->as<Literal>().value() == 1.0 &&
ifFalseExpr->is<Literal>() && ifFalseExpr->as<Literal>().value() == 0.0) {
return ConstructorScalarCast::Make(context, pos, ifTrue->type(), std::move(test));
}
}
return std::make_unique<TernaryExpression>(pos, std::move(test), std::move(ifTrue),
std::move(ifFalse));
}
std::string TernaryExpression::description(OperatorPrecedence parentPrecedence) const {
bool needsParens = (OperatorPrecedence::kTernary >= parentPrecedence);
return std::string(needsParens ? "(" : "") +
this->test()->description(OperatorPrecedence::kTernary) + " ? " +
this->ifTrue()->description(OperatorPrecedence::kTernary) + " : " +
this->ifFalse()->description(OperatorPrecedence::kTernary) +
std::string(needsParens ? ")" : "");
}
} // namespace SkSL