blob: 0b2671df9acdb038cf68f8dd839e19de689bd3b6 [file] [log] [blame]
/*
* Copyright 2020 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/SkSLRehydrator.h"
#include <memory>
#include <unordered_set>
#include "src/sksl/ir/SkSLBinaryExpression.h"
#include "src/sksl/ir/SkSLBreakStatement.h"
#include "src/sksl/ir/SkSLContinueStatement.h"
#include "src/sksl/ir/SkSLDiscardStatement.h"
#include "src/sksl/ir/SkSLDoStatement.h"
#include "src/sksl/ir/SkSLEnum.h"
#include "src/sksl/ir/SkSLExpression.h"
#include "src/sksl/ir/SkSLExpressionStatement.h"
#include "src/sksl/ir/SkSLField.h"
#include "src/sksl/ir/SkSLFieldAccess.h"
#include "src/sksl/ir/SkSLFloatLiteral.h"
#include "src/sksl/ir/SkSLForStatement.h"
#include "src/sksl/ir/SkSLFunctionCall.h"
#include "src/sksl/ir/SkSLFunctionDeclaration.h"
#include "src/sksl/ir/SkSLFunctionDefinition.h"
#include "src/sksl/ir/SkSLIfStatement.h"
#include "src/sksl/ir/SkSLIndexExpression.h"
#include "src/sksl/ir/SkSLInlineMarker.h"
#include "src/sksl/ir/SkSLIntLiteral.h"
#include "src/sksl/ir/SkSLInterfaceBlock.h"
#include "src/sksl/ir/SkSLModifiers.h"
#include "src/sksl/ir/SkSLPostfixExpression.h"
#include "src/sksl/ir/SkSLPrefixExpression.h"
#include "src/sksl/ir/SkSLProgramElement.h"
#include "src/sksl/ir/SkSLReturnStatement.h"
#include "src/sksl/ir/SkSLSetting.h"
#include "src/sksl/ir/SkSLStatement.h"
#include "src/sksl/ir/SkSLStructDefinition.h"
#include "src/sksl/ir/SkSLSwitchCase.h"
#include "src/sksl/ir/SkSLSwitchStatement.h"
#include "src/sksl/ir/SkSLSwizzle.h"
#include "src/sksl/ir/SkSLSymbolAlias.h"
#include "src/sksl/ir/SkSLSymbolTable.h"
#include "src/sksl/ir/SkSLTernaryExpression.h"
#include "src/sksl/ir/SkSLType.h"
#include "src/sksl/ir/SkSLUnresolvedFunction.h"
#include "src/sksl/ir/SkSLVarDeclarations.h"
#include "src/sksl/ir/SkSLVariable.h"
namespace SkSL {
class AutoRehydratorSymbolTable {
public:
AutoRehydratorSymbolTable(Rehydrator* rehydrator)
: fRehydrator(rehydrator)
, fOldSymbols(fRehydrator->fSymbolTable) {
fRehydrator->fSymbolTable = fRehydrator->symbolTable();
}
~AutoRehydratorSymbolTable() {
fRehydrator->fSymbolTable = std::move(fOldSymbols);
}
private:
Rehydrator* fRehydrator;
std::shared_ptr<SymbolTable> fOldSymbols;
};
Rehydrator::Rehydrator(const Context* context, ModifiersPool* modifiers,
std::shared_ptr<SymbolTable> symbolTable, ErrorReporter* errorReporter,
const uint8_t* src, size_t length)
: fContext(*context)
, fModifiers(*modifiers)
, fErrors(errorReporter)
, fSymbolTable(std::move(symbolTable))
, fStart(src)
SkDEBUGCODE(, fEnd(fStart + length)) {
SkASSERT(fSymbolTable);
SkASSERT(fSymbolTable->isBuiltin());
// skip past string data
fIP = fStart;
fIP += this->readU16();
}
Layout Rehydrator::layout() {
switch (this->readU8()) {
case kBuiltinLayout_Command: {
Layout result;
result.fBuiltin = this->readS16();
return result;
}
case kDefaultLayout_Command:
return Layout();
case kLayout_Command: {
int flags = this->readU32();
int location = this->readS8();
int offset = this->readS8();
int binding = this->readS8();
int index = this->readS8();
int set = this->readS8();
int builtin = this->readS16();
int inputAttachmentIndex = this->readS8();
int format = this->readS8();
int primitive = this->readS8();
int maxVertices = this->readS8();
int invocations = this->readS8();
StringFragment marker = this->readString();
StringFragment when = this->readString();
int key = this->readS8();
int ctype = this->readS8();
return Layout(flags, location, offset, binding, index, set, builtin,
inputAttachmentIndex, (Layout::Format) format,
(Layout::Primitive) primitive, maxVertices, invocations, marker, when,
(Layout::Key) key, (Layout::CType) ctype);
}
default:
SkASSERT(false);
return Layout();
}
}
Modifiers Rehydrator::modifiers() {
switch (this->readU8()) {
case kDefaultModifiers_Command:
return Modifiers();
case kModifiers8Bit_Command: {
Layout l = this->layout();
int flags = this->readU8();
return Modifiers(l, flags);
}
case kModifiers_Command: {
Layout l = this->layout();
int flags = this->readS32();
return Modifiers(l, flags);
}
default:
SkASSERT(false);
return Modifiers();
}
}
const Symbol* Rehydrator::symbol() {
int kind = this->readU8();
switch (kind) {
case kArrayType_Command: {
uint16_t id = this->readU16();
const Type* componentType = this->type();
int8_t count = this->readS8();
String name = componentType->name();
if (count == Type::kUnsizedArray) {
name += "[]";
} else {
name += "[" + to_string(count) + "]";
}
const Type* result = fSymbolTable->takeOwnershipOfSymbol(
Type::MakeArrayType(name, *componentType, count));
this->addSymbol(id, result);
return result;
}
case kEnumType_Command: {
uint16_t id = this->readU16();
StringFragment name = this->readString();
const Type* result = fSymbolTable->takeOwnershipOfSymbol(Type::MakeEnumType(name));
this->addSymbol(id, result);
return result;
}
case kFunctionDeclaration_Command: {
uint16_t id = this->readU16();
Modifiers modifiers = this->modifiers();
StringFragment name = this->readString();
int parameterCount = this->readU8();
std::vector<const Variable*> parameters;
parameters.reserve(parameterCount);
for (int i = 0; i < parameterCount; ++i) {
parameters.push_back(this->symbolRef<Variable>(Symbol::Kind::kVariable));
}
const Type* returnType = this->type();
const FunctionDeclaration* result =
fSymbolTable->takeOwnershipOfSymbol(std::make_unique<FunctionDeclaration>(
/*offset=*/-1, fModifiers.addToPool(modifiers), name,
std::move(parameters), returnType, /*builtin=*/true));
this->addSymbol(id, result);
return result;
}
case kField_Command: {
const Variable* owner = this->symbolRef<Variable>(Symbol::Kind::kVariable);
uint8_t index = this->readU8();
const Field* result = fSymbolTable->takeOwnershipOfSymbol(
std::make_unique<Field>(/*offset=*/-1, owner, index));
return result;
}
case kStructType_Command: {
uint16_t id = this->readU16();
StringFragment name = this->readString();
uint8_t fieldCount = this->readU8();
std::vector<Type::Field> fields;
fields.reserve(fieldCount);
for (int i = 0; i < fieldCount; ++i) {
Modifiers m = this->modifiers();
StringFragment fieldName = this->readString();
const Type* type = this->type();
fields.emplace_back(m, fieldName, type);
}
const Type* result = fSymbolTable->takeOwnershipOfSymbol(
Type::MakeStructType(/*offset=*/-1, name, std::move(fields)));
this->addSymbol(id, result);
return result;
}
case kSymbolRef_Command: {
uint16_t id = this->readU16();
SkASSERT(fSymbols.size() > id);
return fSymbols[id];
}
case kSymbolAlias_Command: {
uint16_t id = this->readU16();
StringFragment name = this->readString();
const Symbol* origSymbol = this->symbol();
const SymbolAlias* symbolAlias = fSymbolTable->takeOwnershipOfSymbol(
std::make_unique<SymbolAlias>(/*offset=*/-1, name, origSymbol));
this->addSymbol(id, symbolAlias);
return symbolAlias;
}
case kSystemType_Command: {
uint16_t id = this->readU16();
StringFragment name = this->readString();
const Symbol* result = (*fSymbolTable)[name];
SkASSERT(result && result->kind() == Symbol::Kind::kType);
this->addSymbol(id, result);
return result;
}
case kUnresolvedFunction_Command: {
uint16_t id = this->readU16();
int length = this->readU8();
std::vector<const FunctionDeclaration*> functions;
functions.reserve(length);
for (int i = 0; i < length; ++i) {
const Symbol* f = this->symbol();
SkASSERT(f && f->kind() == Symbol::Kind::kFunctionDeclaration);
functions.push_back((const FunctionDeclaration*) f);
}
const UnresolvedFunction* result = fSymbolTable->takeOwnershipOfSymbol(
std::make_unique<UnresolvedFunction>(std::move(functions)));
this->addSymbol(id, result);
return result;
}
case kVariable_Command: {
uint16_t id = this->readU16();
const Modifiers* m = fModifiers.addToPool(this->modifiers());
StringFragment name = this->readString();
const Type* type = this->type();
Variable::Storage storage = (Variable::Storage) this->readU8();
const Variable* result = fSymbolTable->takeOwnershipOfSymbol(std::make_unique<Variable>(
/*offset=*/-1, m, name, type, /*builtin=*/true, storage));
this->addSymbol(id, result);
return result;
}
default:
printf("unsupported symbol %d\n", kind);
SkASSERT(false);
return nullptr;
}
}
const Type* Rehydrator::type() {
const Symbol* result = this->symbol();
SkASSERT(result->kind() == Symbol::Kind::kType);
return (const Type*) result;
}
std::vector<std::unique_ptr<ProgramElement>> Rehydrator::elements() {
SkDEBUGCODE(uint8_t command = )this->readU8();
SkASSERT(command == kElements_Command);
std::vector<std::unique_ptr<ProgramElement>> result;
while (std::unique_ptr<ProgramElement> elem = this->element()) {
result.push_back(std::move(elem));
}
return result;
}
std::unique_ptr<ProgramElement> Rehydrator::element() {
int kind = this->readU8();
switch (kind) {
case Rehydrator::kEnum_Command: {
StringFragment typeName = this->readString();
std::shared_ptr<SymbolTable> symbols = this->symbolTable(/*inherit=*/false);
for (auto& s : symbols->fOwnedSymbols) {
SkASSERT(s->kind() == Symbol::Kind::kVariable);
Variable& v = (Variable&) *s;
int value = this->readS32();
// enum variables aren't really 'declared', but we have to create a declaration to
// store the value
auto valueLiteral = std::make_unique<IntLiteral>(fContext, /*offset=*/-1, value);
auto declaration = std::make_unique<VarDeclaration>(&v, &v.type(), /*arraySize=*/0,
std::move(valueLiteral));
v.setDeclaration(declaration.get());
symbols->takeOwnershipOfIRNode(std::move(declaration));
}
return std::make_unique<Enum>(/*offset=*/-1, typeName, std::move(symbols),
/*isSharedWithCpp=*/true, /*isBuiltin=*/true);
}
case Rehydrator::kFunctionDefinition_Command: {
const FunctionDeclaration* decl = this->symbolRef<FunctionDeclaration>(
Symbol::Kind::kFunctionDeclaration);
std::unique_ptr<Statement> body = this->statement();
std::unordered_set<const FunctionDeclaration*> refs;
uint8_t refCount = this->readU8();
for (int i = 0; i < refCount; ++i) {
refs.insert(this->symbolRef<FunctionDeclaration>(
Symbol::Kind::kFunctionDeclaration));
}
auto result = std::make_unique<FunctionDefinition>(/*offset=*/-1, decl,
/*builtin=*/true, std::move(body),
std::move(refs));
decl->setDefinition(result.get());
return std::move(result);
}
case Rehydrator::kInterfaceBlock_Command: {
const Symbol* var = this->symbol();
SkASSERT(var && var->is<Variable>());
StringFragment typeName = this->readString();
StringFragment instanceName = this->readString();
int arraySize = this->readS8();
return std::make_unique<InterfaceBlock>(/*offset=*/-1, &var->as<Variable>(), typeName,
instanceName, arraySize, nullptr);
}
case Rehydrator::kVarDeclarations_Command: {
std::unique_ptr<Statement> decl = this->statement();
return std::make_unique<GlobalVarDeclaration>(/*offset=*/-1, std::move(decl));
}
case Rehydrator::kStructDefinition_Command: {
const Symbol* type = this->symbol();
SkASSERT(type && type->is<Type>());
return std::make_unique<StructDefinition>(/*offset=*/-1, type->as<Type>());
}
case Rehydrator::kElementsComplete_Command:
return nullptr;
default:
SkDEBUGFAILF("unsupported element %d\n", kind);
return nullptr;
}
}
std::unique_ptr<Statement> Rehydrator::statement() {
int kind = this->readU8();
switch (kind) {
case Rehydrator::kBlock_Command: {
AutoRehydratorSymbolTable symbols(this);
int count = this->readU8();
StatementArray statements;
statements.reserve_back(count);
for (int i = 0; i < count; ++i) {
statements.push_back(this->statement());
}
bool isScope = this->readU8();
return std::make_unique<Block>(/*offset=*/-1, std::move(statements), fSymbolTable,
isScope);
}
case Rehydrator::kBreak_Command:
return std::unique_ptr<Statement>(new BreakStatement(-1));
case Rehydrator::kContinue_Command:
return std::unique_ptr<Statement>(new ContinueStatement(-1));
case Rehydrator::kDiscard_Command:
return std::unique_ptr<Statement>(new DiscardStatement(-1));
case Rehydrator::kDo_Command: {
std::unique_ptr<Statement> stmt = this->statement();
std::unique_ptr<Expression> expr = this->expression();
return std::unique_ptr<Statement>(new DoStatement(-1, std::move(stmt),
std::move(expr)));
}
case Rehydrator::kExpressionStatement_Command: {
std::unique_ptr<Expression> expr = this->expression();
return std::unique_ptr<Statement>(new ExpressionStatement(std::move(expr)));
}
case Rehydrator::kFor_Command: {
std::unique_ptr<Statement> initializer = this->statement();
std::unique_ptr<Expression> test = this->expression();
std::unique_ptr<Expression> next = this->expression();
std::unique_ptr<Statement> body = this->statement();
std::shared_ptr<SymbolTable> symbols = this->symbolTable();
return std::unique_ptr<Statement>(new ForStatement(-1, std::move(initializer),
std::move(test), std::move(next),
std::move(body),
std::move(symbols)));
}
case Rehydrator::kIf_Command: {
bool isStatic = this->readU8();
std::unique_ptr<Expression> test = this->expression();
std::unique_ptr<Statement> ifTrue = this->statement();
std::unique_ptr<Statement> ifFalse = this->statement();
return std::unique_ptr<Statement>(new IfStatement(-1, isStatic, std::move(test),
std::move(ifTrue),
std::move(ifFalse)));
}
case Rehydrator::kInlineMarker_Command: {
const FunctionDeclaration* funcDecl = this->symbolRef<FunctionDeclaration>(
Symbol::Kind::kFunctionDeclaration);
return std::make_unique<InlineMarker>(funcDecl);
}
case Rehydrator::kReturn_Command: {
std::unique_ptr<Expression> expr = this->expression();
if (expr) {
return std::unique_ptr<Statement>(new ReturnStatement(std::move(expr)));
} else {
return std::unique_ptr<Statement>(new ReturnStatement(-1));
}
}
case Rehydrator::kSwitch_Command: {
bool isStatic = this->readU8();
AutoRehydratorSymbolTable symbols(this);
std::unique_ptr<Expression> expr = this->expression();
int caseCount = this->readU8();
std::vector<std::unique_ptr<SwitchCase>> cases;
cases.reserve(caseCount);
for (int i = 0; i < caseCount; ++i) {
std::unique_ptr<Expression> value = this->expression();
int statementCount = this->readU8();
StatementArray statements;
statements.reserve_back(statementCount);
for (int j = 0; j < statementCount; ++j) {
statements.push_back(this->statement());
}
cases.push_back(std::make_unique<SwitchCase>(/*offset=*/-1, std::move(value),
std::move(statements)));
}
return std::make_unique<SwitchStatement>(-1, isStatic, std::move(expr),
std::move(cases), fSymbolTable);
}
case Rehydrator::kVarDeclaration_Command: {
Variable* var = this->symbolRef<Variable>(Symbol::Kind::kVariable);
const Type* baseType = this->type();
int arraySize = this->readS8();
std::unique_ptr<Expression> value = this->expression();
auto result = std::make_unique<VarDeclaration>(var, baseType, arraySize,
std::move(value));
var->setDeclaration(result.get());
return std::move(result);
}
case Rehydrator::kVoid_Command:
return nullptr;
default:
printf("unsupported statement %d\n", kind);
SkASSERT(false);
return nullptr;
}
}
std::unique_ptr<Expression> Rehydrator::expression() {
int kind = this->readU8();
switch (kind) {
case Rehydrator::kBinary_Command: {
std::unique_ptr<Expression> left = this->expression();
Token::Kind op = (Token::Kind) this->readU8();
std::unique_ptr<Expression> right = this->expression();
const Type* type = this->type();
return std::make_unique<BinaryExpression>(-1, std::move(left), op, std::move(right),
type);
}
case Rehydrator::kBoolLiteral_Command: {
bool value = this->readU8();
return std::make_unique<BoolLiteral>(fContext, -1, value);
}
case Rehydrator::kConstructor_Command: {
const Type* type = this->type();
uint8_t argCount = this->readU8();
ExpressionArray args;
args.reserve_back(argCount);
for (int i = 0; i < argCount; ++i) {
args.push_back(this->expression());
}
return std::make_unique<Constructor>(-1, type, std::move(args));
}
case Rehydrator::kFieldAccess_Command: {
std::unique_ptr<Expression> base = this->expression();
int index = this->readU8();
FieldAccess::OwnerKind ownerKind = (FieldAccess::OwnerKind) this->readU8();
return std::make_unique<FieldAccess>(std::move(base), index, ownerKind);
}
case Rehydrator::kFloatLiteral_Command: {
const Type* type = this->type();
FloatIntUnion u;
u.fInt = this->readS32();
return std::make_unique<FloatLiteral>(-1, u.fFloat, type);
}
case Rehydrator::kFunctionCall_Command: {
const Type* type = this->type();
const FunctionDeclaration* f = this->symbolRef<FunctionDeclaration>(
Symbol::Kind::kFunctionDeclaration);
uint8_t argCount = this->readU8();
ExpressionArray args;
args.reserve_back(argCount);
for (int i = 0; i < argCount; ++i) {
args.push_back(this->expression());
}
return std::make_unique<FunctionCall>(-1, type, f, std::move(args));
}
case Rehydrator::kIndex_Command: {
std::unique_ptr<Expression> base = this->expression();
std::unique_ptr<Expression> index = this->expression();
return std::make_unique<IndexExpression>(fContext, std::move(base), std::move(index));
}
case Rehydrator::kIntLiteral_Command: {
const Type* type = this->type();
int value = this->readS32();
return std::make_unique<IntLiteral>(-1, value, type);
}
case Rehydrator::kPostfix_Command: {
Token::Kind op = (Token::Kind) this->readU8();
std::unique_ptr<Expression> operand = this->expression();
return std::make_unique<PostfixExpression>(std::move(operand), op);
}
case Rehydrator::kPrefix_Command: {
Token::Kind op = (Token::Kind) this->readU8();
std::unique_ptr<Expression> operand = this->expression();
return std::make_unique<PrefixExpression>(op, std::move(operand));
}
case Rehydrator::kSetting_Command: {
StringFragment name = this->readString();
const Type* type = this->type();
return std::make_unique<Setting>(-1, name, type);
}
case Rehydrator::kSwizzle_Command: {
std::unique_ptr<Expression> base = this->expression();
int count = this->readU8();
ComponentArray components;
for (int i = 0; i < count; ++i) {
components.push_back(this->readU8());
}
return std::make_unique<Swizzle>(fContext, std::move(base), components);
}
case Rehydrator::kTernary_Command: {
std::unique_ptr<Expression> test = this->expression();
std::unique_ptr<Expression> ifTrue = this->expression();
std::unique_ptr<Expression> ifFalse = this->expression();
return std::make_unique<TernaryExpression>(-1, std::move(test), std::move(ifTrue),
std::move(ifFalse));
}
case Rehydrator::kVariableReference_Command: {
const Variable* var = this->symbolRef<Variable>(Symbol::Kind::kVariable);
VariableReference::RefKind refKind = (VariableReference::RefKind) this->readU8();
return std::make_unique<VariableReference>(-1, var, refKind);
}
case Rehydrator::kVoid_Command:
return nullptr;
default:
printf("unsupported expression %d\n", kind);
SkASSERT(false);
return nullptr;
}
}
std::shared_ptr<SymbolTable> Rehydrator::symbolTable(bool inherit) {
int command = this->readU8();
if (command == kVoid_Command) {
return nullptr;
}
SkASSERT(command == kSymbolTable_Command);
uint16_t ownedCount = this->readU16();
std::shared_ptr<SymbolTable> oldTable = fSymbolTable;
std::shared_ptr<SymbolTable> result =
inherit ? std::make_shared<SymbolTable>(fSymbolTable, /*builtin=*/true)
: std::make_shared<SymbolTable>(fErrors, /*builtin=*/true);
fSymbolTable = result;
std::vector<const Symbol*> ownedSymbols;
ownedSymbols.reserve(ownedCount);
for (int i = 0; i < ownedCount; ++i) {
ownedSymbols.push_back(this->symbol());
}
uint16_t symbolCount = this->readU16();
std::vector<std::pair<StringFragment, int>> symbols;
symbols.reserve(symbolCount);
for (int i = 0; i < symbolCount; ++i) {
int index = this->readU16();
fSymbolTable->addWithoutOwnership(ownedSymbols[index]);
}
fSymbolTable = oldTable;
return result;
}
} // namespace SkSL