/* 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 /** @file ConstantOptimiser.cpp * @author Christian * @date 2015 */ #include #include #include using namespace solidity; using namespace solidity::evmasm; unsigned ConstantOptimisationMethod::optimiseConstants( bool _isCreation, size_t _runs, langutil::EVMVersion _evmVersion, Assembly& _assembly ) { // TODO: design the optimiser in a way this is not needed AssemblyItems& _items = _assembly.items(); unsigned optimisations = 0; std::map pushes; for (AssemblyItem const& item: _items) if (item.type() == Push) pushes[item]++; std::map pendingReplacements; for (auto it: pushes) { AssemblyItem const& item = it.first; if (item.data() < 0x100) continue; Params params; params.multiplicity = it.second; params.isCreation = _isCreation; params.runs = _runs; params.evmVersion = _evmVersion; LiteralMethod lit(params, item.data()); bigint literalGas = lit.gasNeeded(); CodeCopyMethod copy(params, item.data()); bigint copyGas = copy.gasNeeded(); ComputeMethod compute(params, item.data()); bigint computeGas = compute.gasNeeded(); AssemblyItems replacement; if (copyGas < literalGas && copyGas < computeGas) { replacement = copy.execute(_assembly); optimisations++; } else if (computeGas < literalGas && computeGas <= copyGas) { replacement = compute.execute(_assembly); optimisations++; } if (!replacement.empty()) pendingReplacements[item.data()] = replacement; } if (!pendingReplacements.empty()) replaceConstants(_items, pendingReplacements); return optimisations; } bigint ConstantOptimisationMethod::simpleRunGas(AssemblyItems const& _items, langutil::EVMVersion _evmVersion) { bigint gas = 0; for (AssemblyItem const& item: _items) if (item.type() == Push) gas += GasMeter::pushGas(item.data(), _evmVersion); else if (item.type() == Operation) { if (item.instruction() == Instruction::EXP) gas += GasCosts::expGas; else gas += GasMeter::runGas(item.instruction(), _evmVersion); } return gas; } bigint ConstantOptimisationMethod::dataGas(bytes const& _data) const { assertThrow(_data.size() > 0, OptimizerException, "Empty bytecode generated."); return bigint(GasMeter::dataGas(_data, m_params.isCreation, m_params.evmVersion)); } size_t ConstantOptimisationMethod::bytesRequired(AssemblyItems const& _items, langutil::EVMVersion _evmVersion) { return evmasm::bytesRequired(_items, 3, _evmVersion, Precision::Approximate); // assume 3 byte addresses } void ConstantOptimisationMethod::replaceConstants( AssemblyItems& _items, std::map const& _replacements ) { AssemblyItems replaced; for (AssemblyItem const& item: _items) { if (item.type() == Push) { auto it = _replacements.find(item.data()); if (it != _replacements.end()) { replaced += it->second; continue; } } replaced.push_back(item); } _items = std::move(replaced); } bigint LiteralMethod::gasNeeded() const { return combineGas( simpleRunGas({Instruction::PUSH1}, m_params.evmVersion), // PUSHX plus data (m_params.isCreation ? GasCosts::txDataNonZeroGas(m_params.evmVersion) : GasCosts::createDataGas) + dataGas(toCompactBigEndian(m_value, 1)), 0 ); } bigint CodeCopyMethod::gasNeeded() const { return combineGas( // Run gas: we ignore memory increase costs simpleRunGas(copyRoutine(), m_params.evmVersion) + GasCosts::copyGas, // Data gas for copy routines: Some bytes are zero, but we ignore them. bytesRequired(copyRoutine(), m_params.evmVersion) * (m_params.isCreation ? GasCosts::txDataNonZeroGas(m_params.evmVersion) : GasCosts::createDataGas), // Data gas for data itself dataGas(toBigEndian(m_value)) ); } AssemblyItems CodeCopyMethod::execute(Assembly& _assembly) const { bytes data = toBigEndian(m_value); assertThrow(data.size() == 32, OptimizerException, "Invalid number encoding."); AssemblyItems actualCopyRoutine = copyRoutine(); if (m_params.evmVersion.hasPush0()) actualCopyRoutine[3] = _assembly.newData(data); else actualCopyRoutine[4] = _assembly.newData(data); return actualCopyRoutine; } AssemblyItems const& CodeCopyMethod::copyRoutine() const { // PUSH0 is cheaper than PUSHn/DUP/SWAP. if (m_params.evmVersion.hasPush0()) { // This costs ~29 gas. AssemblyItems static copyRoutine{ // back up memory // mload(0) u256(0), Instruction::MLOAD, // codecopy(0, , 32) u256(32), AssemblyItem(PushData, u256(1) << 16), // replaced above in actualCopyRoutine[3] u256(0), Instruction::CODECOPY, // mload(0) u256(0), Instruction::MLOAD, // restore original memory // mstore(0, x) Instruction::SWAP1, u256(0), Instruction::MSTORE }; return copyRoutine; } else { // This costs ~33 gas. AssemblyItems static copyRoutine{ // constant to be reused 3+ times u256(0), // back up memory // mload(0) Instruction::DUP1, Instruction::MLOAD, // codecopy(0, , 32) u256(32), AssemblyItem(PushData, u256(1) << 16), // replaced above in actualCopyRoutine[4] Instruction::DUP4, Instruction::CODECOPY, // mload(0) Instruction::DUP2, Instruction::MLOAD, // restore original memory // mstore(0, x) Instruction::SWAP2, Instruction::MSTORE }; return copyRoutine; } } AssemblyItems ComputeMethod::findRepresentation(u256 const& _value) { if (_value < 0x10000) // Very small value, not worth computing return AssemblyItems{_value}; else if (numberEncodingSize(~_value) < numberEncodingSize(_value)) // Negated is shorter to represent return findRepresentation(~_value) + AssemblyItems{Instruction::NOT}; else { // Decompose value into a * 2**k + b where abs(b) << 2**k // Is not always better, try literal and decomposition method. AssemblyItems routine{u256(_value)}; bigint bestGas = gasNeeded(routine); for (unsigned bits = 255; bits > 8 && m_maxSteps > 0; --bits) { unsigned gapDetector = unsigned((_value >> (bits - 8)) & 0x1ff); if (gapDetector != 0xff && gapDetector != 0x100) continue; u256 powerOfTwo = u256(1) << bits; u256 upperPart = _value >> bits; bigint lowerPart = _value & (powerOfTwo - 1); if ((powerOfTwo - lowerPart) < lowerPart) { lowerPart = lowerPart - powerOfTwo; // make it negative upperPart++; } if (upperPart == 0) continue; if (abs(lowerPart) >= (powerOfTwo >> 8)) continue; AssemblyItems newRoutine; if (lowerPart != 0) newRoutine += findRepresentation(u256(abs(lowerPart))); if (m_params.evmVersion.hasBitwiseShifting()) { newRoutine += findRepresentation(upperPart); newRoutine += AssemblyItems{u256(bits), Instruction::SHL}; } else { newRoutine += AssemblyItems{u256(bits), u256(2), Instruction::EXP}; if (upperPart != 1) newRoutine += findRepresentation(upperPart) + AssemblyItems{Instruction::MUL}; } if (lowerPart > 0) newRoutine += AssemblyItems{Instruction::ADD}; else if (lowerPart < 0) newRoutine.push_back(Instruction::SUB); if (m_maxSteps > 0) m_maxSteps--; bigint newGas = gasNeeded(newRoutine); if (newGas < bestGas) { bestGas = std::move(newGas); routine = std::move(newRoutine); } } return routine; } } bool ComputeMethod::checkRepresentation(u256 const& _value, AssemblyItems const& _routine) const { // This is a tiny EVM that can only evaluate some instructions. std::vector stack; for (AssemblyItem const& item: _routine) { switch (item.type()) { case Operation: { if (stack.size() < item.arguments()) return false; u256* sp = &stack.back(); switch (item.instruction()) { case Instruction::MUL: sp[-1] = sp[0] * sp[-1]; break; case Instruction::EXP: if (sp[-1] > 0xff) return false; sp[-1] = boost::multiprecision::pow(sp[0], unsigned(sp[-1])); break; case Instruction::ADD: sp[-1] = sp[0] + sp[-1]; break; case Instruction::SUB: sp[-1] = sp[0] - sp[-1]; break; case Instruction::NOT: sp[0] = ~sp[0]; break; case Instruction::SHL: assertThrow( m_params.evmVersion.hasBitwiseShifting(), OptimizerException, "Shift generated for invalid EVM version." ); assertThrow(sp[0] <= u256(255), OptimizerException, "Invalid shift generated."); sp[-1] = u256(bigint(sp[-1]) << unsigned(sp[0])); break; case Instruction::SHR: assertThrow( m_params.evmVersion.hasBitwiseShifting(), OptimizerException, "Shift generated for invalid EVM version." ); assertThrow(sp[0] <= u256(255), OptimizerException, "Invalid shift generated."); sp[-1] = sp[-1] >> unsigned(sp[0]); break; default: return false; } stack.resize(stack.size() + item.deposit()); break; } case Push: stack.push_back(item.data()); break; default: return false; } } return stack.size() == 1 && stack.front() == _value; } bigint ComputeMethod::gasNeeded(AssemblyItems const& _routine) const { auto numExps = static_cast(count(_routine.begin(), _routine.end(), Instruction::EXP)); return combineGas( simpleRunGas(_routine, m_params.evmVersion) + numExps * (GasCosts::expGas + GasCosts::expByteGas(m_params.evmVersion)), // Data gas for routine: Some bytes are zero, but we ignore them. bytesRequired(_routine, m_params.evmVersion) * (m_params.isCreation ? GasCosts::txDataNonZeroGas(m_params.evmVersion) : GasCosts::createDataGas), 0 ); }