mirror of
				https://github.com/ethereum/solidity
				synced 2023-10-03 13:03:40 +00:00 
			
		
		
		
	
		
			
				
	
	
		
			433 lines
		
	
	
		
			12 KiB
		
	
	
	
		
			C++
		
	
	
	
	
	
			
		
		
	
	
			433 lines
		
	
	
		
			12 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
 | |
| 
 | |
| #include <libyul/AST.h>
 | |
| #include <libyul/backends/wasm/WordSizeTransform.h>
 | |
| #include <libyul/Utilities.h>
 | |
| #include <libyul/Dialect.h>
 | |
| #include <libyul/optimiser/NameDisplacer.h>
 | |
| 
 | |
| #include <libsolutil/CommonData.h>
 | |
| 
 | |
| #include <array>
 | |
| #include <map>
 | |
| #include <variant>
 | |
| 
 | |
| using namespace std;
 | |
| using namespace solidity;
 | |
| using namespace solidity::yul;
 | |
| using namespace solidity::util;
 | |
| 
 | |
| void WordSizeTransform::operator()(FunctionDefinition& _fd)
 | |
| {
 | |
| 	rewriteVarDeclList(_fd.parameters);
 | |
| 	rewriteVarDeclList(_fd.returnVariables);
 | |
| 	(*this)(_fd.body);
 | |
| }
 | |
| 
 | |
| void WordSizeTransform::operator()(FunctionCall& _fc)
 | |
| {
 | |
| 	vector<optional<LiteralKind>> const* literalArguments = nullptr;
 | |
| 
 | |
| 	if (BuiltinFunction const* fun = m_inputDialect.builtin(_fc.functionName.name))
 | |
| 		if (!fun->literalArguments.empty())
 | |
| 			literalArguments = &fun->literalArguments;
 | |
| 
 | |
| 	vector<Expression> newArgs;
 | |
| 
 | |
| 	for (size_t i = 0; i < _fc.arguments.size(); i++)
 | |
| 		if (!literalArguments || !(*literalArguments)[i].has_value())
 | |
| 			newArgs += expandValueToVector(_fc.arguments[i]);
 | |
| 		else
 | |
| 		{
 | |
| 			get<Literal>(_fc.arguments[i]).type = m_targetDialect.defaultType;
 | |
| 			newArgs.emplace_back(std::move(_fc.arguments[i]));
 | |
| 		}
 | |
| 
 | |
| 	_fc.arguments = std::move(newArgs);
 | |
| }
 | |
| 
 | |
| void WordSizeTransform::operator()(If& _if)
 | |
| {
 | |
| 	_if.condition = make_unique<Expression>(FunctionCall{
 | |
| 		locationOf(*_if.condition),
 | |
| 		Identifier{locationOf(*_if.condition), "or_bool"_yulstring},
 | |
| 		expandValueToVector(*_if.condition)
 | |
| 	});
 | |
| 	(*this)(_if.body);
 | |
| }
 | |
| 
 | |
| void WordSizeTransform::operator()(Switch&)
 | |
| {
 | |
| 	yulAssert(false, "Switch statement has to be handled inside the containing block.");
 | |
| }
 | |
| 
 | |
| void WordSizeTransform::operator()(ForLoop& _for)
 | |
| {
 | |
| 	(*this)(_for.pre);
 | |
| 	_for.condition = make_unique<Expression>(FunctionCall{
 | |
| 		locationOf(*_for.condition),
 | |
| 		Identifier{locationOf(*_for.condition), "or_bool"_yulstring},
 | |
| 		expandValueToVector(*_for.condition)
 | |
| 	});
 | |
| 	(*this)(_for.post);
 | |
| 	(*this)(_for.body);
 | |
| }
 | |
| 
 | |
| void WordSizeTransform::operator()(Block& _block)
 | |
| {
 | |
| 	iterateReplacing(
 | |
| 		_block.statements,
 | |
| 		[&](Statement& _s) -> std::optional<vector<Statement>>
 | |
| 		{
 | |
| 			if (holds_alternative<VariableDeclaration>(_s))
 | |
| 			{
 | |
| 				VariableDeclaration& varDecl = std::get<VariableDeclaration>(_s);
 | |
| 
 | |
| 				if (!varDecl.value)
 | |
| 					rewriteVarDeclList(varDecl.variables);
 | |
| 				else if (holds_alternative<FunctionCall>(*varDecl.value))
 | |
| 				{
 | |
| 					visit(*varDecl.value);
 | |
| 
 | |
| 					// Special handling for datasize and dataoffset - they will only need one variable.
 | |
| 					if (BuiltinFunction const* f = m_inputDialect.builtin(std::get<FunctionCall>(*varDecl.value).functionName.name))
 | |
| 						if (f->name == "datasize"_yulstring || f->name == "dataoffset"_yulstring)
 | |
| 						{
 | |
| 							yulAssert(f->literalArguments.size() == 1, "");
 | |
| 							yulAssert(f->literalArguments.at(0) == LiteralKind::String, "");
 | |
| 							yulAssert(varDecl.variables.size() == 1, "");
 | |
| 							auto newLhs = generateU64IdentifierNames(varDecl.variables[0].name);
 | |
| 							vector<Statement> ret;
 | |
| 							for (size_t i = 0; i < 3; i++)
 | |
| 								ret.emplace_back(VariableDeclaration{
 | |
| 									varDecl.location,
 | |
| 									{TypedName{varDecl.location, newLhs[i], m_targetDialect.defaultType}},
 | |
| 									make_unique<Expression>(Literal{
 | |
| 										locationOf(*varDecl.value),
 | |
| 										LiteralKind::Number,
 | |
| 										"0"_yulstring,
 | |
| 										m_targetDialect.defaultType
 | |
| 									})
 | |
| 								});
 | |
| 							ret.emplace_back(VariableDeclaration{
 | |
| 								varDecl.location,
 | |
| 								{TypedName{varDecl.location, newLhs[3], m_targetDialect.defaultType}},
 | |
| 								std::move(varDecl.value)
 | |
| 							});
 | |
| 							return {std::move(ret)};
 | |
| 						}
 | |
| 
 | |
| 					rewriteVarDeclList(varDecl.variables);
 | |
| 					return std::nullopt;
 | |
| 				}
 | |
| 				else if (
 | |
| 					holds_alternative<Identifier>(*varDecl.value) ||
 | |
| 					holds_alternative<Literal>(*varDecl.value)
 | |
| 				)
 | |
| 				{
 | |
| 					yulAssert(varDecl.variables.size() == 1, "");
 | |
| 					auto newRhs = expandValue(*varDecl.value);
 | |
| 					auto newLhs = generateU64IdentifierNames(varDecl.variables[0].name);
 | |
| 					vector<Statement> ret;
 | |
| 					for (size_t i = 0; i < 4; i++)
 | |
| 						ret.emplace_back(VariableDeclaration{
 | |
| 								varDecl.location,
 | |
| 								{TypedName{varDecl.location, newLhs[i], m_targetDialect.defaultType}},
 | |
| 								std::move(newRhs[i])
 | |
| 							}
 | |
| 						);
 | |
| 					return {std::move(ret)};
 | |
| 				}
 | |
| 				else
 | |
| 					yulAssert(false, "");
 | |
| 			}
 | |
| 			else if (holds_alternative<Assignment>(_s))
 | |
| 			{
 | |
| 				Assignment& assignment = std::get<Assignment>(_s);
 | |
| 				yulAssert(assignment.value, "");
 | |
| 
 | |
| 				if (holds_alternative<FunctionCall>(*assignment.value))
 | |
| 				{
 | |
| 					visit(*assignment.value);
 | |
| 
 | |
| 					// Special handling for datasize and dataoffset - they will only need one variable.
 | |
| 					if (BuiltinFunction const* f = m_inputDialect.builtin(std::get<FunctionCall>(*assignment.value).functionName.name))
 | |
| 						if (f->name == "datasize"_yulstring || f->name == "dataoffset"_yulstring)
 | |
| 						{
 | |
| 							yulAssert(f->literalArguments.size() == 1, "");
 | |
| 							yulAssert(f->literalArguments[0] == LiteralKind::String, "");
 | |
| 							yulAssert(assignment.variableNames.size() == 1, "");
 | |
| 							auto newLhs = generateU64IdentifierNames(assignment.variableNames[0].name);
 | |
| 							vector<Statement> ret;
 | |
| 							for (size_t i = 0; i < 3; i++)
 | |
| 								ret.emplace_back(Assignment{
 | |
| 									assignment.location,
 | |
| 									{Identifier{assignment.location, newLhs[i]}},
 | |
| 									make_unique<Expression>(Literal{
 | |
| 										locationOf(*assignment.value),
 | |
| 										LiteralKind::Number,
 | |
| 										"0"_yulstring,
 | |
| 										m_targetDialect.defaultType
 | |
| 									})
 | |
| 								});
 | |
| 							ret.emplace_back(Assignment{
 | |
| 								assignment.location,
 | |
| 								{Identifier{assignment.location, newLhs[3]}},
 | |
| 								std::move(assignment.value)
 | |
| 							});
 | |
| 							return {std::move(ret)};
 | |
| 						}
 | |
| 
 | |
| 					rewriteIdentifierList(assignment.variableNames);
 | |
| 					return std::nullopt;
 | |
| 				}
 | |
| 				else if (
 | |
| 					holds_alternative<Identifier>(*assignment.value) ||
 | |
| 					holds_alternative<Literal>(*assignment.value)
 | |
| 				)
 | |
| 				{
 | |
| 					yulAssert(assignment.variableNames.size() == 1, "");
 | |
| 					auto newRhs = expandValue(*assignment.value);
 | |
| 					YulString lhsName = assignment.variableNames[0].name;
 | |
| 					vector<Statement> ret;
 | |
| 					for (size_t i = 0; i < 4; i++)
 | |
| 						ret.emplace_back(Assignment{
 | |
| 								assignment.location,
 | |
| 								{Identifier{assignment.location, m_variableMapping.at(lhsName)[i]}},
 | |
| 								std::move(newRhs[i])
 | |
| 							}
 | |
| 						);
 | |
| 					return {std::move(ret)};
 | |
| 				}
 | |
| 				else
 | |
| 					yulAssert(false, "");
 | |
| 			}
 | |
| 			else if (holds_alternative<Switch>(_s))
 | |
| 				return handleSwitch(std::get<Switch>(_s));
 | |
| 			else
 | |
| 				visit(_s);
 | |
| 			return std::nullopt;
 | |
| 		}
 | |
| 	);
 | |
| }
 | |
| 
 | |
| void WordSizeTransform::run(
 | |
| 	Dialect const& _inputDialect,
 | |
| 	Dialect const& _targetDialect,
 | |
| 	Block& _ast,
 | |
| 	NameDispenser& _nameDispenser
 | |
| )
 | |
| {
 | |
| 	// Free the name `or_bool`.
 | |
| 	NameDisplacer{_nameDispenser, {"or_bool"_yulstring}}(_ast);
 | |
| 	WordSizeTransform{_inputDialect, _targetDialect, _nameDispenser}(_ast);
 | |
| }
 | |
| 
 | |
| WordSizeTransform::WordSizeTransform(
 | |
| 	Dialect const& _inputDialect,
 | |
| 	Dialect const& _targetDialect,
 | |
| 	NameDispenser& _nameDispenser
 | |
| ):
 | |
| 	m_inputDialect(_inputDialect),
 | |
| 	m_targetDialect(_targetDialect),
 | |
| 	m_nameDispenser(_nameDispenser)
 | |
| {
 | |
| }
 | |
| 
 | |
| void WordSizeTransform::rewriteVarDeclList(TypedNameList& _nameList)
 | |
| {
 | |
| 	iterateReplacing(
 | |
| 		_nameList,
 | |
| 		[&](TypedName const& _n) -> std::optional<TypedNameList>
 | |
| 		{
 | |
| 			TypedNameList ret;
 | |
| 			for (auto newName: generateU64IdentifierNames(_n.name))
 | |
| 				ret.emplace_back(TypedName{_n.location, newName, m_targetDialect.defaultType});
 | |
| 			return ret;
 | |
| 		}
 | |
| 	);
 | |
| }
 | |
| 
 | |
| void WordSizeTransform::rewriteIdentifierList(vector<Identifier>& _ids)
 | |
| {
 | |
| 	iterateReplacing(
 | |
| 		_ids,
 | |
| 		[&](Identifier const& _id) -> std::optional<vector<Identifier>>
 | |
| 		{
 | |
| 			vector<Identifier> ret;
 | |
| 			for (auto newId: m_variableMapping.at(_id.name))
 | |
| 				ret.push_back(Identifier{_id.location, newId});
 | |
| 			return ret;
 | |
| 		}
 | |
| 	);
 | |
| }
 | |
| 
 | |
| vector<Statement> WordSizeTransform::handleSwitchInternal(
 | |
| 	langutil::SourceLocation const& _location,
 | |
| 	vector<YulString> const& _splitExpressions,
 | |
| 	vector<Case> _cases,
 | |
| 	YulString _runDefaultFlag,
 | |
| 	size_t _depth
 | |
| )
 | |
| {
 | |
| 	if (_depth == 4)
 | |
| 	{
 | |
| 		yulAssert(_cases.size() == 1, "");
 | |
| 		return std::move(_cases.front().body.statements);
 | |
| 	}
 | |
| 
 | |
| 	// Extract current 64 bit segment and group by it.
 | |
| 	map<u256, vector<Case>> cases;
 | |
| 	for (Case& c: _cases)
 | |
| 	{
 | |
| 		yulAssert(c.value, "Default case still present.");
 | |
| 		cases[
 | |
| 			(valueOfLiteral(*c.value) >> (256 - 64 * (_depth + 1)))	&
 | |
| 			std::numeric_limits<uint64_t>::max()
 | |
| 		].emplace_back(std::move(c));
 | |
| 	}
 | |
| 
 | |
| 	Switch ret{
 | |
| 		_location,
 | |
| 		make_unique<Expression>(Identifier{_location, _splitExpressions.at(_depth)}),
 | |
| 		{}
 | |
| 	};
 | |
| 
 | |
| 	for (auto& c: cases)
 | |
| 	{
 | |
| 		Literal label{_location, LiteralKind::Number, YulString(c.first.str()), m_targetDialect.defaultType};
 | |
| 		ret.cases.emplace_back(Case{
 | |
| 			c.second.front().location,
 | |
| 			make_unique<Literal>(std::move(label)),
 | |
| 			Block{_location, handleSwitchInternal(
 | |
| 				_location,
 | |
| 				_splitExpressions,
 | |
| 				std::move(c.second),
 | |
| 				_runDefaultFlag,
 | |
| 				_depth + 1
 | |
| 			)}
 | |
| 		});
 | |
| 	}
 | |
| 	if (!_runDefaultFlag.empty())
 | |
| 		ret.cases.emplace_back(Case{
 | |
| 			_location,
 | |
| 			nullptr,
 | |
| 			Block{_location, make_vector<Statement>(
 | |
| 				Assignment{
 | |
| 					_location,
 | |
| 					{{_location, _runDefaultFlag}},
 | |
| 					make_unique<Expression>(Literal{_location, LiteralKind::Boolean, "true"_yulstring, m_targetDialect.boolType})
 | |
| 				}
 | |
| 			)}
 | |
| 		});
 | |
| 	return make_vector<Statement>(std::move(ret));
 | |
| }
 | |
| 
 | |
| std::vector<Statement> WordSizeTransform::handleSwitch(Switch& _switch)
 | |
| {
 | |
| 	for (auto& c: _switch.cases)
 | |
| 		(*this)(c.body);
 | |
| 
 | |
| 	// Turns the switch into a quadruply-nested switch plus
 | |
| 	// a flag that tells to execute the default case after all the switches.
 | |
| 	vector<Statement> ret;
 | |
| 
 | |
| 	YulString runDefaultFlag;
 | |
| 	Case defaultCase;
 | |
| 	if (!_switch.cases.back().value)
 | |
| 	{
 | |
| 		runDefaultFlag = m_nameDispenser.newName("run_default"_yulstring);
 | |
| 		defaultCase = std::move(_switch.cases.back());
 | |
| 		_switch.cases.pop_back();
 | |
| 		ret.emplace_back(VariableDeclaration{
 | |
| 			_switch.location,
 | |
| 			{TypedName{_switch.location, runDefaultFlag, m_targetDialect.boolType}},
 | |
| 			{}
 | |
| 		});
 | |
| 	}
 | |
| 	vector<YulString> splitExpressions;
 | |
| 	for (auto const& expr: expandValue(*_switch.expression))
 | |
| 		splitExpressions.emplace_back(std::get<Identifier>(*expr).name);
 | |
| 
 | |
| 	ret += handleSwitchInternal(
 | |
| 		_switch.location,
 | |
| 		splitExpressions,
 | |
| 		std::move(_switch.cases),
 | |
| 		runDefaultFlag,
 | |
| 		0
 | |
| 	);
 | |
| 	if (!runDefaultFlag.empty())
 | |
| 		ret.emplace_back(If{
 | |
| 			_switch.location,
 | |
| 			make_unique<Expression>(Identifier{_switch.location, runDefaultFlag}),
 | |
| 			std::move(defaultCase.body)
 | |
| 		});
 | |
| 	return ret;
 | |
| }
 | |
| 
 | |
| 
 | |
| array<YulString, 4> WordSizeTransform::generateU64IdentifierNames(YulString const& _s)
 | |
| {
 | |
| 	yulAssert(m_variableMapping.find(_s) == m_variableMapping.end(), "");
 | |
| 	for (size_t i = 0; i < 4; i++)
 | |
| 		m_variableMapping[_s][i] = m_nameDispenser.newName(YulString{_s.str() + "_" + to_string(i)});
 | |
| 	return m_variableMapping[_s];
 | |
| }
 | |
| 
 | |
| array<unique_ptr<Expression>, 4> WordSizeTransform::expandValue(Expression const& _e)
 | |
| {
 | |
| 	array<unique_ptr<Expression>, 4> ret;
 | |
| 	if (holds_alternative<Identifier>(_e))
 | |
| 	{
 | |
| 		auto const& id = std::get<Identifier>(_e);
 | |
| 		for (size_t i = 0; i < 4; i++)
 | |
| 			ret[i] = make_unique<Expression>(Identifier{id.location, m_variableMapping.at(id.name)[i]});
 | |
| 	}
 | |
| 	else if (holds_alternative<Literal>(_e))
 | |
| 	{
 | |
| 		auto const& lit = std::get<Literal>(_e);
 | |
| 		u256 val = valueOfLiteral(lit);
 | |
| 		for (size_t exprIndex = 0; exprIndex < 4; ++exprIndex)
 | |
| 		{
 | |
| 			size_t exprIndexReverse = 3 - exprIndex;
 | |
| 			u256 currentVal = val & std::numeric_limits<uint64_t>::max();
 | |
| 			val >>= 64;
 | |
| 			ret[exprIndexReverse] = make_unique<Expression>(
 | |
| 				Literal{
 | |
| 					lit.location,
 | |
| 					LiteralKind::Number,
 | |
| 					YulString(currentVal.str()),
 | |
| 					m_targetDialect.defaultType
 | |
| 				}
 | |
| 			);
 | |
| 		}
 | |
| 	}
 | |
| 	else
 | |
| 		yulAssert(false, "Invalid expression to split.");
 | |
| 	return ret;
 | |
| }
 | |
| 
 | |
| vector<Expression> WordSizeTransform::expandValueToVector(Expression const& _e)
 | |
| {
 | |
| 	vector<Expression> ret;
 | |
| 	for (unique_ptr<Expression>& val: expandValue(_e))
 | |
| 		ret.emplace_back(std::move(*val));
 | |
| 	return ret;
 | |
| }
 |