xref: /aosp_15_r20/external/skia/src/sksl/ir/SkSLExpressionStatement.cpp (revision c8dee2aa9b3f27cf6c858bd81872bdeb2c07ed17)
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