mirror of
https://github.com/ethereum/solidity
synced 2023-10-03 13:03:40 +00:00
155 lines
6.6 KiB
C++
155 lines
6.6 KiB
C++
/*
|
|
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 <http://www.gnu.org/licenses/>.
|
|
*/
|
|
// SPDX-License-Identifier: GPL-3.0
|
|
/**
|
|
* @author Christian <c@ethdev.com>
|
|
* @author Gav Wood <g@ethdev.com>
|
|
* @date 2014
|
|
* Container of the (implicit and explicit) global objects.
|
|
*/
|
|
|
|
#include <libsolidity/analysis/GlobalContext.h>
|
|
|
|
#include <libsolidity/ast/AST.h>
|
|
#include <libsolidity/ast/TypeProvider.h>
|
|
#include <libsolidity/ast/Types.h>
|
|
#include <memory>
|
|
|
|
using namespace std;
|
|
|
|
namespace solidity::frontend
|
|
{
|
|
|
|
namespace
|
|
{
|
|
/// Magic variables get negative ids for easy differentiation
|
|
int magicVariableToID(std::string const& _name)
|
|
{
|
|
if (_name == "abi") return -1;
|
|
else if (_name == "addmod") return -2;
|
|
else if (_name == "assert") return -3;
|
|
else if (_name == "block") return -4;
|
|
else if (_name == "blockhash") return -5;
|
|
else if (_name == "ecrecover") return -6;
|
|
else if (_name == "gasleft") return -7;
|
|
else if (_name == "keccak256") return -8;
|
|
else if (_name == "msg") return -15;
|
|
else if (_name == "mulmod") return -16;
|
|
else if (_name == "now") return -17;
|
|
else if (_name == "require") return -18;
|
|
else if (_name == "revert") return -19;
|
|
else if (_name == "ripemd160") return -20;
|
|
else if (_name == "selfdestruct") return -21;
|
|
else if (_name == "sha256") return -22;
|
|
else if (_name == "sha3") return -23;
|
|
else if (_name == "suicide") return -24;
|
|
else if (_name == "super") return -25;
|
|
else if (_name == "tx") return -26;
|
|
else if (_name == "type") return -27;
|
|
else if (_name == "this") return -28;
|
|
else
|
|
solAssert(false, "Unknown magic variable: \"" + _name + "\".");
|
|
}
|
|
|
|
inline vector<shared_ptr<MagicVariableDeclaration const>> constructMagicVariables()
|
|
{
|
|
static auto const magicVarDecl = [](string const& _name, Type const* _type) {
|
|
return make_shared<MagicVariableDeclaration>(magicVariableToID(_name), _name, _type);
|
|
};
|
|
|
|
return {
|
|
magicVarDecl("abi", TypeProvider::magic(MagicType::Kind::ABI)),
|
|
magicVarDecl("addmod", TypeProvider::function(strings{"uint256", "uint256", "uint256"}, strings{"uint256"}, FunctionType::Kind::AddMod, StateMutability::Pure)),
|
|
magicVarDecl("assert", TypeProvider::function(strings{"bool"}, strings{}, FunctionType::Kind::Assert, StateMutability::Pure)),
|
|
magicVarDecl("block", TypeProvider::magic(MagicType::Kind::Block)),
|
|
magicVarDecl("blockhash", TypeProvider::function(strings{"uint256"}, strings{"bytes32"}, FunctionType::Kind::BlockHash, StateMutability::View)),
|
|
magicVarDecl("ecrecover", TypeProvider::function(strings{"bytes32", "uint8", "bytes32", "bytes32"}, strings{"address"}, FunctionType::Kind::ECRecover, StateMutability::Pure)),
|
|
magicVarDecl("gasleft", TypeProvider::function(strings(), strings{"uint256"}, FunctionType::Kind::GasLeft, StateMutability::View)),
|
|
magicVarDecl("keccak256", TypeProvider::function(strings{"bytes memory"}, strings{"bytes32"}, FunctionType::Kind::KECCAK256, StateMutability::Pure)),
|
|
magicVarDecl("msg", TypeProvider::magic(MagicType::Kind::Message)),
|
|
magicVarDecl("mulmod", TypeProvider::function(strings{"uint256", "uint256", "uint256"}, strings{"uint256"}, FunctionType::Kind::MulMod, StateMutability::Pure)),
|
|
magicVarDecl("now", TypeProvider::uint256()),
|
|
magicVarDecl("require", TypeProvider::function(strings{"bool"}, strings{}, FunctionType::Kind::Require, StateMutability::Pure)),
|
|
magicVarDecl("require", TypeProvider::function(strings{"bool", "string memory"}, strings{}, FunctionType::Kind::Require, StateMutability::Pure)),
|
|
magicVarDecl("revert", TypeProvider::function(strings(), strings(), FunctionType::Kind::Revert, StateMutability::Pure)),
|
|
magicVarDecl("revert", TypeProvider::function(strings{"string memory"}, strings(), FunctionType::Kind::Revert, StateMutability::Pure)),
|
|
magicVarDecl("ripemd160", TypeProvider::function(strings{"bytes memory"}, strings{"bytes20"}, FunctionType::Kind::RIPEMD160, StateMutability::Pure)),
|
|
magicVarDecl("selfdestruct", TypeProvider::function(strings{"address payable"}, strings{}, FunctionType::Kind::Selfdestruct)),
|
|
magicVarDecl("sha256", TypeProvider::function(strings{"bytes memory"}, strings{"bytes32"}, FunctionType::Kind::SHA256, StateMutability::Pure)),
|
|
magicVarDecl("sha3", TypeProvider::function(strings{"bytes memory"}, strings{"bytes32"}, FunctionType::Kind::KECCAK256, StateMutability::Pure)),
|
|
magicVarDecl("suicide", TypeProvider::function(strings{"address payable"}, strings{}, FunctionType::Kind::Selfdestruct)),
|
|
magicVarDecl("tx", TypeProvider::magic(MagicType::Kind::Transaction)),
|
|
// Accepts a MagicType that can be any contract type or an Integer type and returns a
|
|
// MagicType. The TypeChecker handles the correctness of the input and output types.
|
|
magicVarDecl("type", TypeProvider::function(
|
|
strings{},
|
|
strings{},
|
|
FunctionType::Kind::MetaType,
|
|
StateMutability::Pure,
|
|
FunctionType::Options::withArbitraryParameters()
|
|
)),
|
|
};
|
|
}
|
|
|
|
}
|
|
|
|
GlobalContext::GlobalContext(): m_magicVariables{constructMagicVariables()}
|
|
{
|
|
}
|
|
|
|
void GlobalContext::setCurrentContract(ContractDefinition const& _contract)
|
|
{
|
|
m_currentContract = &_contract;
|
|
}
|
|
|
|
vector<Declaration const*> GlobalContext::declarations() const
|
|
{
|
|
vector<Declaration const*> declarations;
|
|
declarations.reserve(m_magicVariables.size());
|
|
for (ASTPointer<MagicVariableDeclaration const> const& variable: m_magicVariables)
|
|
declarations.push_back(variable.get());
|
|
return declarations;
|
|
}
|
|
|
|
MagicVariableDeclaration const* GlobalContext::currentThis() const
|
|
{
|
|
if (!m_thisPointer[m_currentContract])
|
|
{
|
|
Type const* type = TypeProvider::emptyTuple();
|
|
if (m_currentContract)
|
|
type = TypeProvider::contract(*m_currentContract);
|
|
m_thisPointer[m_currentContract] =
|
|
make_shared<MagicVariableDeclaration>(magicVariableToID("this"), "this", type);
|
|
}
|
|
return m_thisPointer[m_currentContract].get();
|
|
}
|
|
|
|
MagicVariableDeclaration const* GlobalContext::currentSuper() const
|
|
{
|
|
if (!m_superPointer[m_currentContract])
|
|
{
|
|
Type const* type = TypeProvider::emptyTuple();
|
|
if (m_currentContract)
|
|
type = TypeProvider::typeType(TypeProvider::contract(*m_currentContract, true));
|
|
m_superPointer[m_currentContract] =
|
|
make_shared<MagicVariableDeclaration>(magicVariableToID("super"), "super", type);
|
|
}
|
|
return m_superPointer[m_currentContract].get();
|
|
}
|
|
|
|
}
|