/* 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 . */ /** * Generator for code that handles LValues. */ #include #include #include #include #include #include using namespace std; using namespace dev; using namespace dev::solidity; IRLocalVariable::IRLocalVariable( IRGenerationContext& _context, VariableDeclaration const& _varDecl ): IRLValue(_context, _varDecl.annotation().type), m_variableName(_context.localVariableName(_varDecl)) { } string IRLocalVariable::storeValue(string const& _value, Type const& _type) const { solAssert(_type == *m_type, "Storing different types - not necessarily a problem."); return m_variableName + " := " + _value + "\n"; } string IRLocalVariable::setToZero() const { return storeValue(m_context.utils().zeroValueFunction(*m_type) + "()", *m_type); } IRStorageItem::IRStorageItem( IRGenerationContext& _context, VariableDeclaration const& _varDecl ): IRStorageItem( _context, *_varDecl.annotation().type, _context.storageLocationOfVariable(_varDecl) ) { } IRStorageItem::IRStorageItem( IRGenerationContext& _context, Type const& _type, std::pair slot_offset ): IRLValue(_context, &_type), m_slot(toCompactHexWithPrefix(slot_offset.first)), m_offset(slot_offset.second) { } IRStorageItem::IRStorageItem( IRGenerationContext& _context, string _slot, boost::variant _offset, Type const& _type ): IRLValue(_context, &_type), m_slot(move(_slot)), m_offset(std::move(_offset)) { solAssert(!m_offset.empty(), ""); solAssert(!m_slot.empty(), ""); } string IRStorageItem::retrieveValue() const { if (!m_type->isValueType()) return m_slot; solUnimplementedAssert(m_type->category() != Type::Category::Function, ""); if (m_offset.type() == typeid(string)) return m_context.utils().readFromStorageDynamic(*m_type, false) + "(" + m_slot + ", " + boost::get(m_offset) + ")"; else if (m_offset.type() == typeid(unsigned)) return m_context.utils().readFromStorage(*m_type, boost::get(m_offset), false) + "(" + m_slot + ")"; solAssert(false, ""); } string IRStorageItem::storeValue(string const& _value, Type const& _sourceType) const { if (m_type->isValueType()) solAssert(_sourceType == *m_type, "Different type, but might not be an error."); boost::optional offset; if (m_offset.type() == typeid(unsigned)) offset = get(m_offset); return m_context.utils().updateStorageValueFunction(*m_type, offset) + "(" + m_slot + (m_offset.type() == typeid(string) ? (", " + get(m_offset)) : "" ) + ", " + _value + ")\n"; } string IRStorageItem::setToZero() const { solUnimplementedAssert(m_type->isValueType(), ""); return storeValue(m_context.utils().zeroValueFunction(*m_type) + "()", *m_type); } IRStorageArrayLength::IRStorageArrayLength(IRGenerationContext& _context, string _slot, Type const& _type, ArrayType const& _arrayType): IRLValue(_context, &_type), m_arrayType(_arrayType), m_slot(move(_slot)) { solAssert(*m_type == *TypeProvider::uint256(), "Must be uint256!"); } string IRStorageArrayLength::retrieveValue() const { return m_context.utils().arrayLengthFunction(m_arrayType) + "(" + m_slot + ")\n"; } string IRStorageArrayLength::storeValue(std::string const& _value, Type const& _type) const { solAssert(_type == *m_type, "Different type, but might not be an error."); return m_context.utils().resizeDynamicArrayFunction(m_arrayType) + "(" + m_slot + ", " + _value + ")\n"; } string IRStorageArrayLength::setToZero() const { return storeValue("0", *TypeProvider::uint256()); }