1*c8dee2aaSAndroid Build Coastguard Worker /*
2*c8dee2aaSAndroid Build Coastguard Worker * Copyright 2021 Google LLC
3*c8dee2aaSAndroid Build Coastguard Worker *
4*c8dee2aaSAndroid Build Coastguard Worker * Use of this source code is governed by a BSD-style license that can be
5*c8dee2aaSAndroid Build Coastguard Worker * found in the LICENSE file.
6*c8dee2aaSAndroid Build Coastguard Worker */
7*c8dee2aaSAndroid Build Coastguard Worker
8*c8dee2aaSAndroid Build Coastguard Worker #include "src/sksl/ir/SkSLExpressionStatement.h"
9*c8dee2aaSAndroid Build Coastguard Worker
10*c8dee2aaSAndroid Build Coastguard Worker #include "include/core/SkTypes.h"
11*c8dee2aaSAndroid Build Coastguard Worker #include "src/sksl/SkSLAnalysis.h"
12*c8dee2aaSAndroid Build Coastguard Worker #include "src/sksl/SkSLContext.h"
13*c8dee2aaSAndroid Build Coastguard Worker #include "src/sksl/SkSLOperator.h"
14*c8dee2aaSAndroid Build Coastguard Worker #include "src/sksl/SkSLProgramSettings.h"
15*c8dee2aaSAndroid Build Coastguard Worker #include "src/sksl/ir/SkSLBinaryExpression.h"
16*c8dee2aaSAndroid Build Coastguard Worker #include "src/sksl/ir/SkSLNop.h"
17*c8dee2aaSAndroid Build Coastguard Worker #include "src/sksl/ir/SkSLVariableReference.h"
18*c8dee2aaSAndroid Build Coastguard Worker
19*c8dee2aaSAndroid Build Coastguard Worker namespace SkSL {
20*c8dee2aaSAndroid Build Coastguard Worker
Convert(const Context & context,std::unique_ptr<Expression> expr)21*c8dee2aaSAndroid Build Coastguard Worker std::unique_ptr<Statement> ExpressionStatement::Convert(const Context& context,
22*c8dee2aaSAndroid Build Coastguard Worker std::unique_ptr<Expression> expr) {
23*c8dee2aaSAndroid Build Coastguard Worker // Expression-statements need to represent a complete expression.
24*c8dee2aaSAndroid Build Coastguard Worker // Report an error on intermediate expressions, like FunctionReference or TypeReference.
25*c8dee2aaSAndroid Build Coastguard Worker if (expr->isIncomplete(context)) {
26*c8dee2aaSAndroid Build Coastguard Worker return nullptr;
27*c8dee2aaSAndroid Build Coastguard Worker }
28*c8dee2aaSAndroid Build Coastguard Worker return ExpressionStatement::Make(context, std::move(expr));
29*c8dee2aaSAndroid Build Coastguard Worker }
30*c8dee2aaSAndroid Build Coastguard Worker
Make(const Context & context,std::unique_ptr<Expression> expr)31*c8dee2aaSAndroid Build Coastguard Worker std::unique_ptr<Statement> ExpressionStatement::Make(const Context& context,
32*c8dee2aaSAndroid Build Coastguard Worker std::unique_ptr<Expression> expr) {
33*c8dee2aaSAndroid Build Coastguard Worker SkASSERT(!expr->isIncomplete(context));
34*c8dee2aaSAndroid Build Coastguard Worker
35*c8dee2aaSAndroid Build Coastguard Worker if (context.fConfig->fSettings.fOptimize) {
36*c8dee2aaSAndroid Build Coastguard Worker // Expression-statements without any side effect can be replaced with a Nop.
37*c8dee2aaSAndroid Build Coastguard Worker if (!Analysis::HasSideEffects(*expr)) {
38*c8dee2aaSAndroid Build Coastguard Worker return Nop::Make();
39*c8dee2aaSAndroid Build Coastguard Worker }
40*c8dee2aaSAndroid Build Coastguard Worker
41*c8dee2aaSAndroid Build Coastguard Worker // If this is an assignment statement like `a += b;`, the ref-kind of `a` will be set as
42*c8dee2aaSAndroid Build Coastguard Worker // read-write; `a` is written-to by the +=, and read-from by the consumer of the expression.
43*c8dee2aaSAndroid Build Coastguard Worker // We can demote the ref-kind to "write" safely, because the result of the expression is
44*c8dee2aaSAndroid Build Coastguard Worker // discarded; that is, `a` is never actually read-from.
45*c8dee2aaSAndroid Build Coastguard Worker if (expr->is<BinaryExpression>()) {
46*c8dee2aaSAndroid Build Coastguard Worker BinaryExpression& binary = expr->as<BinaryExpression>();
47*c8dee2aaSAndroid Build Coastguard Worker if (VariableReference* assignedVar = binary.isAssignmentIntoVariable()) {
48*c8dee2aaSAndroid Build Coastguard Worker if (assignedVar->refKind() == VariableRefKind::kReadWrite) {
49*c8dee2aaSAndroid Build Coastguard Worker assignedVar->setRefKind(VariableRefKind::kWrite);
50*c8dee2aaSAndroid Build Coastguard Worker }
51*c8dee2aaSAndroid Build Coastguard Worker }
52*c8dee2aaSAndroid Build Coastguard Worker }
53*c8dee2aaSAndroid Build Coastguard Worker }
54*c8dee2aaSAndroid Build Coastguard Worker
55*c8dee2aaSAndroid Build Coastguard Worker return std::make_unique<ExpressionStatement>(std::move(expr));
56*c8dee2aaSAndroid Build Coastguard Worker }
57*c8dee2aaSAndroid Build Coastguard Worker
description() const58*c8dee2aaSAndroid Build Coastguard Worker std::string ExpressionStatement::description() const {
59*c8dee2aaSAndroid Build Coastguard Worker return this->expression()->description(OperatorPrecedence::kStatement) + ";";
60*c8dee2aaSAndroid Build Coastguard Worker }
61*c8dee2aaSAndroid Build Coastguard Worker
62*c8dee2aaSAndroid Build Coastguard Worker } // namespace SkSL
63