mirror of
https://github.com/ethereum/solidity
synced 2023-10-03 13:03:40 +00:00
Structural gas estimator.
This commit is contained in:
parent
b9d7387e7a
commit
4d62c463d1
104
GasMeter.cpp
Normal file
104
GasMeter.cpp
Normal file
@ -0,0 +1,104 @@
|
||||
/*
|
||||
This file is part of cpp-ethereum.
|
||||
|
||||
cpp-ethereum 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.
|
||||
|
||||
cpp-ethereum 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 cpp-ethereum. If not, see <http://www.gnu.org/licenses/>.
|
||||
*/
|
||||
/** @file GasMeter.cpp
|
||||
* @author Christian <c@ethdev.com>
|
||||
* @date 2015
|
||||
*/
|
||||
|
||||
#include "GasMeter.h"
|
||||
#include <libevmcore/Params.h>
|
||||
|
||||
using namespace std;
|
||||
using namespace dev;
|
||||
using namespace dev::eth;
|
||||
|
||||
GasMeter::GasConsumption& GasMeter::GasConsumption::operator+=(GasConsumption const& _other)
|
||||
{
|
||||
isInfinite = isInfinite || _other.isInfinite;
|
||||
if (isInfinite)
|
||||
return *this;
|
||||
bigint v = bigint(value) + _other.value;
|
||||
if (v > std::numeric_limits<u256>::max())
|
||||
isInfinite = true;
|
||||
else
|
||||
value = u256(v);
|
||||
return *this;
|
||||
}
|
||||
|
||||
GasMeter::GasConsumption GasMeter::estimateMax(AssemblyItem const& _item)
|
||||
{
|
||||
switch (_item.type()) {
|
||||
case Push:
|
||||
case PushTag:
|
||||
return runGas(Instruction::PUSH1);
|
||||
case Tag:
|
||||
return runGas(Instruction::JUMPDEST);
|
||||
case Operation:
|
||||
{
|
||||
GasConsumption gas = runGas(_item.instruction());
|
||||
switch (_item.instruction())
|
||||
{
|
||||
case Instruction::SSTORE:
|
||||
// @todo logic can be improved
|
||||
gas += c_sstoreSetGas;
|
||||
break;
|
||||
case Instruction::SLOAD:
|
||||
gas += c_sloadGas;
|
||||
break;
|
||||
case Instruction::MSTORE:
|
||||
case Instruction::MSTORE8:
|
||||
case Instruction::MLOAD:
|
||||
case Instruction::RETURN:
|
||||
case Instruction::SHA3:
|
||||
case Instruction::CALLDATACOPY:
|
||||
case Instruction::CODECOPY:
|
||||
case Instruction::EXTCODECOPY:
|
||||
case Instruction::LOG0:
|
||||
case Instruction::LOG1:
|
||||
case Instruction::LOG2:
|
||||
case Instruction::LOG3:
|
||||
case Instruction::LOG4:
|
||||
case Instruction::CALL:
|
||||
case Instruction::CALLCODE:
|
||||
case Instruction::CREATE:
|
||||
case Instruction::EXP:
|
||||
// @todo logic can be improved
|
||||
gas = GasConsumption::infinite();
|
||||
break;
|
||||
default:
|
||||
break;
|
||||
}
|
||||
return gas;
|
||||
break;
|
||||
}
|
||||
default:
|
||||
break;
|
||||
}
|
||||
|
||||
return GasConsumption::infinite();
|
||||
}
|
||||
|
||||
GasMeter::GasConsumption GasMeter::runGas(Instruction _instruction)
|
||||
{
|
||||
if (_instruction == Instruction::JUMPDEST)
|
||||
return GasConsumption(1);
|
||||
|
||||
int tier = instructionInfo(_instruction).gasPriceTier;
|
||||
return tier == InvalidTier ? GasConsumption::infinite() : c_tierStepGas[tier];
|
||||
}
|
||||
|
||||
|
67
GasMeter.h
Normal file
67
GasMeter.h
Normal file
@ -0,0 +1,67 @@
|
||||
/*
|
||||
This file is part of cpp-ethereum.
|
||||
|
||||
cpp-ethereum 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.
|
||||
|
||||
cpp-ethereum 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 cpp-ethereum. If not, see <http://www.gnu.org/licenses/>.
|
||||
*/
|
||||
/** @file GasMeter.cpp
|
||||
* @author Christian <c@ethdev.com>
|
||||
* @date 2015
|
||||
*/
|
||||
|
||||
#pragma once
|
||||
|
||||
#include <ostream>
|
||||
#include <libevmasm/AssemblyItem.h>
|
||||
|
||||
namespace dev
|
||||
{
|
||||
namespace eth
|
||||
{
|
||||
|
||||
/**
|
||||
* Class that helps computing the maximum gas consumption for instructions.
|
||||
*/
|
||||
class GasMeter
|
||||
{
|
||||
public:
|
||||
struct GasConsumption
|
||||
{
|
||||
GasConsumption(u256 _value = 0, bool _infinite = false): value(_value), isInfinite(_infinite) {}
|
||||
static GasConsumption infinite() { return GasConsumption(0, true); }
|
||||
|
||||
GasConsumption& operator+=(GasConsumption const& _otherS);
|
||||
std::ostream& operator<<(std::ostream& _str) const;
|
||||
|
||||
u256 value;
|
||||
bool isInfinite;
|
||||
};
|
||||
|
||||
/// Returns an upper bound on the gas consumed by the given instruction.
|
||||
GasConsumption estimateMax(AssemblyItem const& _item);
|
||||
|
||||
private:
|
||||
static GasConsumption runGas(Instruction _instruction);
|
||||
};
|
||||
|
||||
inline std::ostream& operator<<(std::ostream& _str, GasMeter::GasConsumption const& _consumption)
|
||||
{
|
||||
if (_consumption.isInfinite)
|
||||
return _str << "inf";
|
||||
else
|
||||
return _str << _consumption.value;
|
||||
}
|
||||
|
||||
|
||||
}
|
||||
}
|
Loading…
Reference in New Issue
Block a user