/* * Copyright 2016 Google Inc. * * Use of this source code is governed by a BSD-style license that can be * found in the LICENSE file. */ #include "SkSLSymbolTable.h" #include "SkSLUnresolvedFunction.h" namespace SkSL { std::vector SymbolTable::GetFunctions(const Symbol& s) { switch (s.fKind) { case Symbol::kFunctionDeclaration_Kind: return { &((FunctionDeclaration&) s) }; case Symbol::kUnresolvedFunction_Kind: return ((UnresolvedFunction&) s).fFunctions; default: return std::vector(); } } const Symbol* SymbolTable::operator[](StringFragment name) { const auto& entry = fSymbols.find(name); if (entry == fSymbols.end()) { if (fParent) { return (*fParent)[name]; } return nullptr; } if (fParent) { auto functions = GetFunctions(*entry->second); if (functions.size() > 0) { bool modified = false; const Symbol* previous = (*fParent)[name]; if (previous) { auto previousFunctions = GetFunctions(*previous); for (const FunctionDeclaration* prev : previousFunctions) { bool found = false; for (const FunctionDeclaration* current : functions) { if (current->matches(*prev)) { found = true; break; } } if (!found) { functions.push_back(prev); modified = true; } } if (modified) { SkASSERT(functions.size() > 1); return this->takeOwnership(new UnresolvedFunction(functions)); } } } } return entry->second; } Symbol* SymbolTable::takeOwnership(Symbol* s) { fOwnedSymbols.emplace_back(s); return s; } IRNode* SymbolTable::takeOwnership(IRNode* n) { fOwnedNodes.emplace_back(n); return n; } void SymbolTable::add(StringFragment name, std::unique_ptr symbol) { this->addWithoutOwnership(name, symbol.get()); this->takeOwnership(symbol.release()); } void SymbolTable::addWithoutOwnership(StringFragment name, const Symbol* symbol) { const auto& existing = fSymbols.find(name); if (existing == fSymbols.end()) { fSymbols[name] = symbol; } else if (symbol->fKind == Symbol::kFunctionDeclaration_Kind) { const Symbol* oldSymbol = existing->second; if (oldSymbol->fKind == Symbol::kFunctionDeclaration_Kind) { std::vector functions; functions.push_back((const FunctionDeclaration*) oldSymbol); functions.push_back((const FunctionDeclaration*) symbol); UnresolvedFunction* u = new UnresolvedFunction(std::move(functions)); fSymbols[name] = u; this->takeOwnership(u); } else if (oldSymbol->fKind == Symbol::kUnresolvedFunction_Kind) { std::vector functions; for (const auto* f : ((UnresolvedFunction&) *oldSymbol).fFunctions) { functions.push_back(f); } functions.push_back((const FunctionDeclaration*) symbol); UnresolvedFunction* u = new UnresolvedFunction(std::move(functions)); fSymbols[name] = u; this->takeOwnership(u); } } else { fErrorReporter.error(symbol->fOffset, "symbol '" + name + "' was already defined"); } } void SymbolTable::markAllFunctionsBuiltin() { for (const auto& pair : fSymbols) { switch (pair.second->fKind) { case Symbol::kFunctionDeclaration_Kind: ((FunctionDeclaration&) *pair.second).fBuiltin = true; break; case Symbol::kUnresolvedFunction_Kind: for (auto& f : ((UnresolvedFunction&) *pair.second).fFunctions) { ((FunctionDeclaration*) f)->fBuiltin = true; } break; default: break; } } } std::map::iterator SymbolTable::begin() { return fSymbols.begin(); } std::map::iterator SymbolTable::end() { return fSymbols.end(); } } // namespace