blob: ab238af1b405de361721063455abc05ac2d17026 [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.
*/
#ifndef SKSL_THREADCONTEXT
#define SKSL_THREADCONTEXT
#include "include/private/SkSLModifiers.h"
#include "src/sksl/SkSLMangler.h"
#include "src/sksl/ir/SkSLProgram.h"
#if !defined(SKSL_STANDALONE) && SK_SUPPORT_GPU
#include "src/gpu/GrFragmentProcessor.h"
#endif // !defined(SKSL_STANDALONE) && SK_SUPPORT_GPU
#include <list>
#include <stack>
#include <string_view>
namespace SkSL {
class Compiler;
class Context;
struct ParsedModule;
class ProgramElement;
class SymbolTable;
class Type;
class Variable;
namespace dsl {
class DSLCore;
class DSLWriter;
} // namespace dsl
/**
* Thread-safe class that tracks per-thread state associated with SkSL output.
*/
class ThreadContext {
public:
ThreadContext(SkSL::Compiler* compiler, SkSL::ProgramKind kind,
const SkSL::ProgramSettings& settings, SkSL::ParsedModule module, bool isModule);
~ThreadContext();
/**
* Returns true if the DSL has been started.
*/
static bool IsActive();
/**
* Returns the Compiler used by DSL operations in the current thread.
*/
static SkSL::Compiler& Compiler() { return *Instance().fCompiler; }
/**
* Returns the Context used by DSL operations in the current thread.
*/
static SkSL::Context& Context();
/**
* Returns the Settings used by DSL operations in the current thread.
*/
static const SkSL::ProgramSettings& Settings();
/**
* Returns the Program::Inputs used by the current thread.
*/
static SkSL::Program::Inputs& Inputs() { return Instance().fInputs; }
/**
* Returns the collection to which DSL program elements in this thread should be appended.
*/
static std::vector<std::unique_ptr<SkSL::ProgramElement>>& ProgramElements() {
return Instance().fProgramElements;
}
static std::vector<const ProgramElement*>& SharedElements() {
return Instance().fSharedElements;
}
/**
* Returns the current SymbolTable.
*/
static std::shared_ptr<SkSL::SymbolTable>& SymbolTable();
/**
* Returns the current memory pool.
*/
static std::unique_ptr<Pool>& MemoryPool() { return Instance().fPool; }
/**
* Returns the current modifiers pool.
*/
static std::unique_ptr<ModifiersPool>& GetModifiersPool() { return Instance().fModifiersPool; }
/**
* Returns the current ProgramConfig.
*/
static const std::unique_ptr<ProgramConfig>& GetProgramConfig() { return Instance().fConfig; }
static bool IsModule() { return GetProgramConfig()->fIsBuiltinCode; }
/**
* Returns the final pointer to a pooled Modifiers object that should be used to represent the
* given modifiers.
*/
static const SkSL::Modifiers* Modifiers(const SkSL::Modifiers& modifiers);
struct RTAdjustData {
// Points to a standalone sk_RTAdjust variable, if one exists.
const Variable* fVar = nullptr;
// Points to the interface block containing an sk_RTAdjust field, if one exists.
const Variable* fInterfaceBlock = nullptr;
// If fInterfaceBlock is non-null, contains the index of the sk_RTAdjust field within it.
int fFieldIndex = -1;
};
/**
* Returns a struct containing information about the RTAdjust variable.
*/
static RTAdjustData& RTAdjustState();
#if !defined(SKSL_STANDALONE) && SK_SUPPORT_GPU
/**
* Returns the fragment processor for which DSL output is being generated for the current
* thread.
*/
static GrFragmentProcessor::ProgramImpl* CurrentProcessor() {
SkASSERTF(!Instance().fStack.empty(), "This feature requires a FragmentProcessor");
return Instance().fStack.top().fProcessor;
}
/**
* Returns the EmitArgs for fragment processor output in the current thread.
*/
static GrFragmentProcessor::ProgramImpl::EmitArgs* CurrentEmitArgs() {
SkASSERTF(!Instance().fStack.empty(), "This feature requires a FragmentProcessor");
return Instance().fStack.top().fEmitArgs;
}
static bool InFragmentProcessor() {
return !Instance().fStack.empty();
}
/**
* Pushes a new processor / emitArgs pair for the current thread.
*/
static void StartFragmentProcessor(GrFragmentProcessor::ProgramImpl* processor,
GrFragmentProcessor::ProgramImpl::EmitArgs* emitArgs);
/**
* Pops the processor / emitArgs pair associated with the current thread.
*/
static void EndFragmentProcessor();
#else
static bool InFragmentProcessor() {
return false;
}
#endif // !defined(SKSL_STANDALONE) && SK_SUPPORT_GPU
/**
* Returns the ErrorReporter associated with the current thread. This object will be notified
* when any DSL errors occur.
*/
static ErrorReporter& GetErrorReporter() {
return *Context().fErrors;
}
static void SetErrorReporter(ErrorReporter* errorReporter);
/**
* Notifies the current ErrorReporter that an error has occurred. The default error handler
* prints the message to stderr and aborts.
*/
static void ReportError(std::string_view msg, PositionInfo info = PositionInfo::Capture());
/**
* Forwards any pending errors to the DSL ErrorReporter.
*/
static void ReportErrors(PositionInfo pos);
static ThreadContext& Instance();
static void SetInstance(std::unique_ptr<ThreadContext> instance);
private:
class DefaultErrorReporter : public ErrorReporter {
void handleError(std::string_view msg, PositionInfo pos) override;
};
void setupSymbolTable();
std::unique_ptr<SkSL::ProgramConfig> fConfig;
std::unique_ptr<SkSL::ModifiersPool> fModifiersPool;
SkSL::Compiler* fCompiler;
std::unique_ptr<Pool> fPool;
SkSL::ProgramConfig* fOldConfig;
SkSL::ModifiersPool* fOldModifiersPool;
std::vector<std::unique_ptr<SkSL::ProgramElement>> fProgramElements;
std::vector<const SkSL::ProgramElement*> fSharedElements;
DefaultErrorReporter fDefaultErrorReporter;
ErrorReporter& fOldErrorReporter;
ProgramSettings fSettings;
Mangler fMangler;
RTAdjustData fRTAdjust;
Program::Inputs fInputs;
#if !defined(SKSL_STANDALONE) && SK_SUPPORT_GPU
struct StackFrame {
GrFragmentProcessor::ProgramImpl* fProcessor;
GrFragmentProcessor::ProgramImpl::EmitArgs* fEmitArgs;
SkSL::StatementArray fSavedDeclarations;
};
std::stack<StackFrame, std::list<StackFrame>> fStack;
#endif // !defined(SKSL_STANDALONE) && SK_SUPPORT_GPU
friend class dsl::DSLCore;
friend class dsl::DSLWriter;
};
} // namespace SkSL
#endif