1*c8dee2aaSAndroid Build Coastguard Worker /* 2*c8dee2aaSAndroid Build Coastguard Worker * Copyright 2022 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 "include/core/SkSpan.h" 9*c8dee2aaSAndroid Build Coastguard Worker #include "src/sksl/SkSLDefines.h" 10*c8dee2aaSAndroid Build Coastguard Worker #include "src/sksl/SkSLModule.h" 11*c8dee2aaSAndroid Build Coastguard Worker #include "src/sksl/ir/SkSLBlock.h" 12*c8dee2aaSAndroid Build Coastguard Worker #include "src/sksl/ir/SkSLFunctionDefinition.h" 13*c8dee2aaSAndroid Build Coastguard Worker #include "src/sksl/ir/SkSLProgramElement.h" 14*c8dee2aaSAndroid Build Coastguard Worker #include "src/sksl/ir/SkSLStatement.h" 15*c8dee2aaSAndroid Build Coastguard Worker #include "src/sksl/transform/SkSLProgramWriter.h" 16*c8dee2aaSAndroid Build Coastguard Worker #include "src/sksl/transform/SkSLTransform.h" 17*c8dee2aaSAndroid Build Coastguard Worker 18*c8dee2aaSAndroid Build Coastguard Worker #include <algorithm> 19*c8dee2aaSAndroid Build Coastguard Worker #include <iterator> 20*c8dee2aaSAndroid Build Coastguard Worker #include <memory> 21*c8dee2aaSAndroid Build Coastguard Worker #include <vector> 22*c8dee2aaSAndroid Build Coastguard Worker 23*c8dee2aaSAndroid Build Coastguard Worker namespace SkSL { 24*c8dee2aaSAndroid Build Coastguard Worker 25*c8dee2aaSAndroid Build Coastguard Worker class Expression; 26*c8dee2aaSAndroid Build Coastguard Worker eliminate_empty_statements(SkSpan<std::unique_ptr<ProgramElement>> elements)27*c8dee2aaSAndroid Build Coastguard Workerstatic void eliminate_empty_statements(SkSpan<std::unique_ptr<ProgramElement>> elements) { 28*c8dee2aaSAndroid Build Coastguard Worker class EmptyStatementEliminator : public ProgramWriter { 29*c8dee2aaSAndroid Build Coastguard Worker public: 30*c8dee2aaSAndroid Build Coastguard Worker bool visitExpressionPtr(std::unique_ptr<Expression>& expr) override { 31*c8dee2aaSAndroid Build Coastguard Worker // We don't need to look inside expressions at all. 32*c8dee2aaSAndroid Build Coastguard Worker return false; 33*c8dee2aaSAndroid Build Coastguard Worker } 34*c8dee2aaSAndroid Build Coastguard Worker 35*c8dee2aaSAndroid Build Coastguard Worker bool visitStatementPtr(std::unique_ptr<Statement>& stmt) override { 36*c8dee2aaSAndroid Build Coastguard Worker // Work from the innermost blocks to the outermost. 37*c8dee2aaSAndroid Build Coastguard Worker INHERITED::visitStatementPtr(stmt); 38*c8dee2aaSAndroid Build Coastguard Worker 39*c8dee2aaSAndroid Build Coastguard Worker if (stmt->is<Block>()) { 40*c8dee2aaSAndroid Build Coastguard Worker StatementArray& children = stmt->as<Block>().children(); 41*c8dee2aaSAndroid Build Coastguard Worker auto iter = std::remove_if(children.begin(), children.end(), 42*c8dee2aaSAndroid Build Coastguard Worker [](std::unique_ptr<Statement>& stmt) { 43*c8dee2aaSAndroid Build Coastguard Worker return stmt->isEmpty(); 44*c8dee2aaSAndroid Build Coastguard Worker }); 45*c8dee2aaSAndroid Build Coastguard Worker children.resize(std::distance(children.begin(), iter)); 46*c8dee2aaSAndroid Build Coastguard Worker } 47*c8dee2aaSAndroid Build Coastguard Worker 48*c8dee2aaSAndroid Build Coastguard Worker // We always check the entire program. 49*c8dee2aaSAndroid Build Coastguard Worker return false; 50*c8dee2aaSAndroid Build Coastguard Worker } 51*c8dee2aaSAndroid Build Coastguard Worker 52*c8dee2aaSAndroid Build Coastguard Worker using INHERITED = ProgramWriter; 53*c8dee2aaSAndroid Build Coastguard Worker }; 54*c8dee2aaSAndroid Build Coastguard Worker 55*c8dee2aaSAndroid Build Coastguard Worker for (std::unique_ptr<ProgramElement>& pe : elements) { 56*c8dee2aaSAndroid Build Coastguard Worker if (pe->is<FunctionDefinition>()) { 57*c8dee2aaSAndroid Build Coastguard Worker EmptyStatementEliminator visitor; 58*c8dee2aaSAndroid Build Coastguard Worker visitor.visitStatementPtr(pe->as<FunctionDefinition>().body()); 59*c8dee2aaSAndroid Build Coastguard Worker } 60*c8dee2aaSAndroid Build Coastguard Worker } 61*c8dee2aaSAndroid Build Coastguard Worker } 62*c8dee2aaSAndroid Build Coastguard Worker EliminateEmptyStatements(Module & module)63*c8dee2aaSAndroid Build Coastguard Workervoid Transform::EliminateEmptyStatements(Module& module) { 64*c8dee2aaSAndroid Build Coastguard Worker return eliminate_empty_statements(SkSpan(module.fElements)); 65*c8dee2aaSAndroid Build Coastguard Worker } 66*c8dee2aaSAndroid Build Coastguard Worker 67*c8dee2aaSAndroid Build Coastguard Worker } // namespace SkSL 68