blob: dab72806f7e7ee7d0e20ec79e05168d2af1f3717 [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/SkSLAnalysis.h"
#include "include/private/SkSLProgramElement.h"
#include "include/private/SkSLStatement.h"
#include "src/core/SkSafeMath.h"
#include "src/sksl/SkSLContext.h"
#include "src/sksl/SkSLProgramSettings.h"
#include "src/sksl/analysis/SkSLProgramVisitor.h"
#include "src/sksl/ir/SkSLForStatement.h"
#include "src/sksl/ir/SkSLFunctionCall.h"
#include "src/sksl/ir/SkSLFunctionDefinition.h"
#include "src/sksl/ir/SkSLProgram.h"
#include <unordered_map>
#include <vector>
namespace SkSL {
bool Analysis::CheckProgramUnrolledSize(const Program& program) {
// We check the size of strict-ES2 programs since SkVM will completely unroll them.
// Note that we *cannot* safely check the program size of non-ES2 code at this time, as it is
// allowed to do things we can't measure (e.g. the program can contain a recursive cycle). We
// could, at best, compute a lower bound.
const Context& context = *program.fContext;
SkASSERT(context.fConfig->strictES2Mode());
// If we decide that expressions are cheaper than statements, or that certain statements are
// more expensive than others, etc., we can always tweak these ratios as needed. A very rough
// ballpark estimate is currently good enough for our purposes.
static constexpr size_t kExpressionCost = 1;
static constexpr size_t kStatementCost = 1;
static constexpr size_t kUnknownCost = -1;
static constexpr size_t kProgramSizeLimit = 100000;
static constexpr size_t kProgramStackDepthLimit = 50;
class ProgramSizeVisitor : public ProgramVisitor {
public:
ProgramSizeVisitor(const Context& c) : fContext(c) {}
using ProgramVisitor::visitProgramElement;
size_t functionSize() const {
return fFunctionSize;
}
bool visitProgramElement(const ProgramElement& pe) override {
if (pe.is<FunctionDefinition>()) {
// Check the function-size cache map first. We don't need to visit this function if
// we already processed it before.
const FunctionDeclaration* decl = &pe.as<FunctionDefinition>().declaration();
#ifndef SKIA_STRUCTURED_BINDINGS_BACKPORT
auto [iter, wasInserted] = fFunctionCostMap.insert({decl, kUnknownCost});
#else
auto pair = fFunctionCostMap.insert({decl, kUnknownCost});
auto iter = pair.first;
bool wasInserted = pair.second;
#endif
if (!wasInserted) {
// We already have this function in our map. We don't need to check it again.
if (iter->second == kUnknownCost) {
// If the function is present in the map with an unknown cost, we're
// recursively processing it--in other words, we found a cycle in the code.
// Unwind our stack into a string.
String msg = "\n\t" + decl->description();
for (auto unwind = fStack.rbegin(); unwind != fStack.rend(); ++unwind) {
msg = "\n\t" + (*unwind)->description() + msg;
if (*unwind == decl) {
break;
}
}
msg = "potential recursion (function call cycle) not allowed:" + msg;
fContext.fErrors->error(pe.fLine, std::move(msg));
fFunctionSize = iter->second = 0;
return true;
}
// Set the size to its known value.
fFunctionSize = iter->second;
return false;
}
// If the function-call stack has gotten too deep, stop the analysis.
if (fStack.size() >= kProgramStackDepthLimit) {
String msg = "exceeded max function call depth:";
for (auto unwind = fStack.begin(); unwind != fStack.end(); ++unwind) {
msg += "\n\t" + (*unwind)->description();
}
msg += "\n\t" + decl->description();
fContext.fErrors->error(pe.fLine, std::move(msg));
fFunctionSize = iter->second = 0;
return true;
}
// Calculate the function cost and store it in our cache.
fStack.push_back(decl);
fFunctionSize = 0;
bool result = INHERITED::visitProgramElement(pe);
iter->second = fFunctionSize;
fStack.pop_back();
return result;
}
return INHERITED::visitProgramElement(pe);
}
bool visitStatement(const Statement& stmt) override {
switch (stmt.kind()) {
case Statement::Kind::kFor: {
// We count a for-loop's unrolled size here. We expect that the init statement
// will be emitted once, and the next-expr and statement will be repeated in the
// output for every iteration of the loop. The test-expr is optimized away
// during the unroll and is not counted at all.
const ForStatement& forStmt = stmt.as<ForStatement>();
bool result = this->visitStatement(*forStmt.initializer());
size_t originalFunctionSize = fFunctionSize;
fFunctionSize = 0;
result = this->visitExpression(*forStmt.next()) ||
this->visitStatement(*forStmt.statement()) || result;
if (const LoopUnrollInfo* unrollInfo = forStmt.unrollInfo()) {
fFunctionSize = SkSafeMath::Mul(fFunctionSize, unrollInfo->fCount);
} else {
SkDEBUGFAIL("for-loops should always have unroll info in an ES2 program");
}
fFunctionSize = SkSafeMath::Add(fFunctionSize, originalFunctionSize);
return result;
}
case Statement::Kind::kExpression:
// The cost of an expression-statement is counted in visitExpression. It would
// be double-dipping to count it here too.
break;
case Statement::Kind::kInlineMarker:
case Statement::Kind::kNop:
case Statement::Kind::kVarDeclaration:
// These statements don't directly consume any space in a compiled program.
break;
case Statement::Kind::kDo:
SkDEBUGFAIL("encountered a statement that shouldn't exist in an ES2 program");
break;
default:
fFunctionSize = SkSafeMath::Add(fFunctionSize, kStatementCost);
break;
}
bool earlyExit = fFunctionSize > kProgramSizeLimit;
return earlyExit || INHERITED::visitStatement(stmt);
}
bool visitExpression(const Expression& expr) override {
// Other than function calls, all expressions are assumed to have a fixed unit cost.
bool earlyExit = false;
size_t expressionCost = kExpressionCost;
if (expr.is<FunctionCall>()) {
// Visit this function call to calculate its size. If we've already sized it, this
// will retrieve the size from our cache.
const FunctionCall& call = expr.as<FunctionCall>();
const FunctionDeclaration* decl = &call.function();
if (decl->definition() && !decl->isIntrinsic()) {
size_t originalFunctionSize = fFunctionSize;
fFunctionSize = 0;
earlyExit = this->visitProgramElement(*decl->definition());
expressionCost = fFunctionSize;
fFunctionSize = originalFunctionSize;
}
}
fFunctionSize = SkSafeMath::Add(fFunctionSize, expressionCost);
return earlyExit || INHERITED::visitExpression(expr);
}
private:
using INHERITED = ProgramVisitor;
const Context& fContext;
size_t fFunctionSize = 0;
std::unordered_map<const FunctionDeclaration*, size_t> fFunctionCostMap;
std::vector<const FunctionDeclaration*> fStack;
};
// Process every function in our program.
ProgramSizeVisitor visitor{context};
for (const std::unique_ptr<ProgramElement>& element : program.fOwnedElements) {
if (element->is<FunctionDefinition>()) {
// Visit every function--we want to detect static recursion and report it as an error,
// even in unreferenced functions.
visitor.visitProgramElement(*element);
// Report an error when main()'s flattened size is larger than our program limit.
if (visitor.functionSize() > kProgramSizeLimit &&
element->as<FunctionDefinition>().declaration().isMain()) {
context.fErrors->error(/*line=*/-1, "program is too large");
}
}
}
return true;
}
} // namespace SkSL