/*
* Copyright 2017 Google Inc.
*
* Use of this source code is governed by a BSD-style license that can be
* found in the LICENSE file.
*/
#ifndef SKSL_CPPCODEGENERATOR
#define SKSL_CPPCODEGENERATOR
#include "SkSLGLSLCodeGenerator.h"
#include "SkSLSectionAndParameterHelper.h"
#include <set>
namespace SkSL {
class CPPCodeGenerator : public GLSLCodeGenerator {
public:
CPPCodeGenerator(const Context* context, const Program* program, ErrorReporter* errors,
String name, OutputStream* out);
bool generateCode() override;
private:
void writef(const char* s, va_list va) SKSL_PRINTF_LIKE(2, 0);
void writef(const char* s, ...) SKSL_PRINTF_LIKE(2, 3);
bool writeSection(const char* name, const char* prefix = "");
void writeHeader() override;
bool usesPrecisionModifiers() const override;
String getTypeName(const Type& type) override;
void writeBinaryExpression(const BinaryExpression& b, Precedence parentPrecedence) override;
void writeIndexExpression(const IndexExpression& i) override;
void writeIntLiteral(const IntLiteral& i) override;
void writeSwizzle(const Swizzle& swizzle) override;
void writeFieldAccess(const FieldAccess& access) override;
void writeVariableReference(const VariableReference& ref) override;
String getSamplerHandle(const Variable& var);
void writeIfStatement(const IfStatement& s) override;
void writeReturnStatement(const ReturnStatement& s) override;
void writeSwitchStatement(const SwitchStatement& s) override;
void writeFunctionCall(const FunctionCall& c) override;
void writeFunction(const FunctionDefinition& f) override;
void writeSetting(const Setting& s) override;
void writeProgramElement(const ProgramElement& p) override;
void addUniform(const Variable& var);
// writes a printf escape that will be filled in at runtime by the given C++ expression string
void writeRuntimeValue(const Type& type, const Layout& layout, const String& cppCode);
void writeVarInitializer(const Variable& var, const Expression& value) override;
void writeInputVars() override;
void writePrivateVars();
void writePrivateVarValues();
void writeCodeAppend(const String& code);
bool writeEmitCode(std::vector<const Variable*>& uniforms);
void writeSetData(std::vector<const Variable*>& uniforms);
void writeGetKey();
void writeOnTextureSampler();
void writeClone();
void writeTest();
// If the returned C++ is included in the generated code, then the variable name stored in
// cppVar will refer to a valid SkString that matches the Expression. Successful returns leave
// the output buffer (and related state) unmodified.
//
// In the simplest cases, this will return "SkString {cppVar}(\"{e}\");", while more advanced
// cases will properly insert format arguments.
String convertSKSLExpressionToCPP(const Expression& e, const String& cppVar);
// Process accumulated sksl to split it into appended code sections, properly interleaved with
// the extra emit code blocks, based on statement/block locations and the inserted tokens
// from newExtraEmitCodeBlock(). It is necessary to split the sksl after the program has been
// fully walked since many elements redirect fOut to simultaneously build header sections and
// bodies that are then concatenated; due to this it is not possible to split the sksl emission
// on the fly.
void flushEmittedCode();
// Start a new extra emit code block for accumulating C++ code. This will insert a token into
// the sksl stream to mark the fence between previous complete sksl statements and where the
// C++ code added to the new block will be added to emitCode(). These tokens are removed by
// flushEmittedCode() as it consumes them before passing pure sksl to writeCodeAppend().
void newExtraEmitCodeBlock();
// Append CPP code to the current extra emit code block.
void addExtraEmitCodeLine(const String& toAppend);
int getChildFPIndex(const VariableReference& reference) const;
String fName;
String fFullName;
SectionAndParameterHelper fSectionAndParameterHelper;
std::vector<String> fExtraEmitCodeBlocks;
std::vector<String> fFormatArgs;
std::set<int> fWrittenTransformedCoords;
// if true, we are writing a C++ expression instead of a GLSL expression
bool fCPPMode = false;
bool fInMain = false;
// if not null, we are accumulating SkSL for emitCode into fOut, which
// replaced the original buffer with a StringStream. The original buffer is
// stored here for restoration.
OutputStream* fCPPBuffer = nullptr;
typedef GLSLCodeGenerator INHERITED;
};
}
#endif