/* This file is part of solidity. solidity is free software: you can redistribute it and/or modify it under the terms of the GNU General Public License as published by the Free Software Foundation, either version 3 of the License, or (at your option) any later version. solidity is distributed in the hope that it will be useful, but WITHOUT ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License for more details. You should have received a copy of the GNU General Public License along with solidity. If not, see . */ // SPDX-License-Identifier: GPL-3.0 #include #include #include #include #include #include #include using namespace solidity; using namespace solidity::frontend::experimental; using namespace solidity::util; template Type getType(IRGenerationContext const& _context, Node const& _node) { auto& annotation = _context.analysis.annotation(_node); solAssert(annotation.type); return _context.env->resolve(*annotation.type); } namespace { size_t getTypeStackSlots(IRGenerationContext const& _context, Type _type) { } } IRVariable::IRVariable(IRGenerationContext const& _context, std::string _baseName, Type _type): m_baseName(std::move(_baseName)), m_type(_type) { } IRVariable::IRVariable(IRGenerationContext const& _context, VariableDeclaration const& _declaration): IRVariable(_context, IRNames::localVariable(_declaration), getType(_context, _declaration)) { } IRVariable::IRVariable(IRGenerationContext const& _context, Expression const& _expression): IRVariable(_context, IRNames::localVariable(_expression), getType(_context, _expression)) { } IRVariable IRVariable::part(string const& _name) const { for (auto const& [itemName, itemType]: m_type.stackItems()) if (itemName == _name) { solAssert(itemName.empty() || itemType, ""); return IRVariable{suffixedName(itemName), itemType ? *itemType : m_type}; } solAssert(false, "Invalid stack item name: " + _name); } bool IRVariable::hasPart(std::string const& _name) const { for (auto const& [itemName, itemType]: m_type.stackItems()) if (itemName == _name) { solAssert(itemName.empty() || itemType, ""); return true; } return false; } vector IRVariable::stackSlots() const { vector result; for (auto const& [itemName, itemType]: m_type.stackItems()) if (itemType) { solAssert(!itemName.empty(), ""); solAssert(m_type != *itemType, ""); result += IRVariable{suffixedName(itemName), *itemType}.stackSlots(); } else { solAssert(itemName.empty(), ""); result.emplace_back(m_baseName); } return result; } string IRVariable::commaSeparatedList() const { return joinHumanReadable(stackSlots()); } string IRVariable::commaSeparatedListPrefixed() const { return joinHumanReadablePrefixed(stackSlots()); } string IRVariable::name() const { solAssert(m_type.sizeOnStack() == 1, ""); auto const& [itemName, type] = m_type.stackItems().front(); solAssert(!type, "Expected null type for name " + itemName); return suffixedName(itemName); } IRVariable IRVariable::tupleComponent(size_t _i) const { solAssert( m_type.category() == Type::Category::Tuple, "Requested tuple component of non-tuple IR variable." ); return part(IRNames::tupleComponent(_i)); } string IRVariable::suffixedName(string const& _suffix) const { if (_suffix.empty()) return m_baseName; else return m_baseName + '_' + _suffix; }