mirror of
				https://github.com/ethereum/solidity
				synced 2023-10-03 13:03:40 +00:00 
			
		
		
		
	Basic infrastructure.
This commit is contained in:
		
							parent
							
								
									9bce5f91dc
								
							
						
					
					
						commit
						fa815764dd
					
				| @ -46,6 +46,8 @@ set(sources | ||||
| 	analysis/TypeChecker.h | ||||
| 	analysis/ViewPureChecker.cpp | ||||
| 	analysis/ViewPureChecker.h | ||||
| 	analysis/experimental/Analysis.cpp | ||||
| 	analysis/experimental/Analysis.h | ||||
| 	ast/AST.cpp | ||||
| 	ast/AST.h | ||||
| 	ast/AST_accept.h | ||||
| @ -90,6 +92,8 @@ set(sources | ||||
| 	codegen/ReturnInfo.cpp | ||||
| 	codegen/YulUtilFunctions.h | ||||
| 	codegen/YulUtilFunctions.cpp | ||||
| 	codegen/experimental/IRGenerator.cpp | ||||
| 	codegen/experimental/IRGenerator.h | ||||
| 	codegen/ir/Common.cpp | ||||
| 	codegen/ir/Common.h | ||||
| 	codegen/ir/IRGenerator.cpp | ||||
|  | ||||
							
								
								
									
										26
									
								
								libsolidity/analysis/experimental/Analysis.cpp
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										26
									
								
								libsolidity/analysis/experimental/Analysis.cpp
									
									
									
									
									
										Normal file
									
								
							| @ -0,0 +1,26 @@ | ||||
| /*
 | ||||
| 	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 <libsolidity/analysis/experimental/Analysis.h> | ||||
| 
 | ||||
| using namespace solidity::langutil; | ||||
| using namespace solidity::frontend::experimental; | ||||
| 
 | ||||
| bool Analysis::check(ASTNode const&) | ||||
| { | ||||
| 	return true; | ||||
| } | ||||
							
								
								
									
										43
									
								
								libsolidity/analysis/experimental/Analysis.h
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										43
									
								
								libsolidity/analysis/experimental/Analysis.h
									
									
									
									
									
										Normal file
									
								
							| @ -0,0 +1,43 @@ | ||||
| /*
 | ||||
| 	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
 | ||||
| #pragma once | ||||
| 
 | ||||
| namespace solidity::frontend | ||||
| { | ||||
| class ASTNode; | ||||
| } | ||||
| 
 | ||||
| namespace solidity::langutil | ||||
| { | ||||
| class ErrorReporter; | ||||
| } | ||||
| 
 | ||||
| namespace solidity::frontend::experimental | ||||
| { | ||||
| 
 | ||||
| class Analysis | ||||
| { | ||||
| public: | ||||
| 	Analysis(langutil::ErrorReporter& _errorReporter): m_errorReporter(_errorReporter) | ||||
| 	{} | ||||
| 	bool check(ASTNode const& _ast); | ||||
| private: | ||||
| 	langutil::ErrorReporter& m_errorReporter; | ||||
| }; | ||||
| 
 | ||||
| } | ||||
							
								
								
									
										160
									
								
								libsolidity/codegen/experimental/IRGenerator.cpp
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										160
									
								
								libsolidity/codegen/experimental/IRGenerator.cpp
									
									
									
									
									
										Normal file
									
								
							| @ -0,0 +1,160 @@ | ||||
| /*
 | ||||
| 	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 <libsolidity/codegen/experimental/IRGenerator.h> | ||||
| 
 | ||||
| #include <libsolidity/codegen/ir/Common.h> | ||||
| 
 | ||||
| #include <libyul/YulStack.h> | ||||
| #include <libyul/AsmPrinter.h> | ||||
| #include <libyul/AST.h> | ||||
| #include <libyul/optimiser/ASTCopier.h> | ||||
| 
 | ||||
| #include <liblangutil/SourceReferenceFormatter.h> | ||||
| 
 | ||||
| #include <libsolutil/Whiskers.h> | ||||
| 
 | ||||
| #include <variant> | ||||
| 
 | ||||
| using namespace std; | ||||
| using namespace solidity; | ||||
| using namespace solidity::frontend::experimental; | ||||
| using namespace solidity::langutil; | ||||
| using namespace solidity::util; | ||||
| 
 | ||||
| string IRGenerator::run( | ||||
| 	ContractDefinition const& _contract, | ||||
| 	bytes const& /*_cborMetadata*/, | ||||
| 	map<ContractDefinition const*, string_view const> const& /*_otherYulSources*/ | ||||
| ) const | ||||
| { | ||||
| 
 | ||||
| 	Whiskers t(R"( | ||||
| 		object "<CreationObject>" { | ||||
| 			code { | ||||
| 				codecopy(0, dataoffset("<DeployedObject>"), datasize("<DeployedObject>")) | ||||
| 				return(0, datasize("<DeployedObject>")) | ||||
| 			} | ||||
| 			object "<DeployedObject>" { | ||||
| 				code { | ||||
| 					<code> | ||||
| 				} | ||||
| 			} | ||||
| 		} | ||||
| 	)"); | ||||
| 	t("CreationObject", IRNames::creationObject(_contract)); | ||||
| 	t("DeployedObject", IRNames::deployedObject(_contract)); | ||||
| 	t("code", generate(_contract)); | ||||
| 
 | ||||
| 	return t.render(); | ||||
| } | ||||
| 
 | ||||
| string IRGenerator::generate(ContractDefinition const& _contract) const | ||||
| { | ||||
| 	std::stringstream code; | ||||
| 	code << "{\n"; | ||||
| 	if (_contract.fallbackFunction()) | ||||
| 	{ | ||||
| 		code << IRNames::function(*_contract.fallbackFunction()) << "()\n"; | ||||
| 	} | ||||
| 	code << "revert(0,0)\n"; | ||||
| 	code << "}\n"; | ||||
| 
 | ||||
| 	for (FunctionDefinition const* f: _contract.definedFunctions()) | ||||
| 		code << generate(*f); | ||||
| 
 | ||||
| 	return code.str(); | ||||
| } | ||||
| 
 | ||||
| string IRGenerator::generate(FunctionDefinition const& _function) const | ||||
| { | ||||
| 	std::stringstream code; | ||||
| 	code << "function " << IRNames::function(_function) << "() {\n"; | ||||
| 	for (auto _statement: _function.body().statements()) | ||||
| 	{ | ||||
| 		if (auto assembly = dynamic_cast<InlineAssembly const*>(_statement.get())) | ||||
| 			code << generate(*assembly) << "\n"; | ||||
| 		else | ||||
| 			solUnimplemented("Unsupported statement type."); | ||||
| 	} | ||||
| 	code << "}\n"; | ||||
| 	return code.str(); | ||||
| } | ||||
| 
 | ||||
| namespace { | ||||
| 
 | ||||
| struct CopyTranslate: public yul::ASTCopier | ||||
| { | ||||
| 	CopyTranslate( | ||||
| 		yul::Dialect const& _dialect, | ||||
| 		map<yul::Identifier const*, void*> _references | ||||
| 	): m_dialect(_dialect), m_references(std::move(_references)) {} | ||||
| 
 | ||||
| 	using ASTCopier::operator(); | ||||
| 
 | ||||
| 	yul::Expression operator()(yul::Identifier const& _identifier) override | ||||
| 	{ | ||||
| 		// The operator() function is only called in lvalue context. In rvalue context,
 | ||||
| 		// only translate(yul::Identifier) is called.
 | ||||
| 		if (m_references.count(&_identifier)) | ||||
| 			return translateReference(_identifier); | ||||
| 		else | ||||
| 			return ASTCopier::operator()(_identifier); | ||||
| 	} | ||||
| 
 | ||||
| 	yul::YulString translateIdentifier(yul::YulString _name) override | ||||
| 	{ | ||||
| 		if (m_dialect.builtin(_name)) | ||||
| 			return _name; | ||||
| 		else | ||||
| 			return yul::YulString{"usr$" + _name.str()}; | ||||
| 	} | ||||
| 
 | ||||
| 	yul::Identifier translate(yul::Identifier const& _identifier) override | ||||
| 	{ | ||||
| 		if (!m_references.count(&_identifier)) | ||||
| 			return ASTCopier::translate(_identifier); | ||||
| 
 | ||||
| 		yul::Expression translated = translateReference(_identifier); | ||||
| 		solAssert(holds_alternative<yul::Identifier>(translated)); | ||||
| 		return get<yul::Identifier>(std::move(translated)); | ||||
| 	} | ||||
| 
 | ||||
| private: | ||||
| 
 | ||||
| 	/// Translates a reference to a local variable, potentially including
 | ||||
| 	/// a suffix. Might return a literal, which causes this to be invalid in
 | ||||
| 	/// lvalue-context.
 | ||||
| 	yul::Expression translateReference(yul::Identifier const&) | ||||
| 	{ | ||||
| 		solUnimplemented("External references in inline assembly not implemented."); | ||||
| 	} | ||||
| 
 | ||||
| 	yul::Dialect const& m_dialect; | ||||
| 	map<yul::Identifier const*, void*> m_references; | ||||
| }; | ||||
| 
 | ||||
| } | ||||
| 
 | ||||
| string IRGenerator::generate(InlineAssembly const& _assembly) const | ||||
| { | ||||
| 	CopyTranslate bodyCopier{_assembly.dialect(), {}}; | ||||
| 	yul::Statement modified = bodyCopier(_assembly.operations()); | ||||
| 	solAssert(holds_alternative<yul::Block>(modified)); | ||||
| 	return yul::AsmPrinter()(std::get<yul::Block>(modified)); | ||||
| } | ||||
							
								
								
									
										73
									
								
								libsolidity/codegen/experimental/IRGenerator.h
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										73
									
								
								libsolidity/codegen/experimental/IRGenerator.h
									
									
									
									
									
										Normal file
									
								
							| @ -0,0 +1,73 @@ | ||||
| /*
 | ||||
| 	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
 | ||||
| 
 | ||||
| #pragma once | ||||
| 
 | ||||
| #include <libsolidity/interface/DebugSettings.h> | ||||
| #include <libsolidity/interface/OptimiserSettings.h> | ||||
| #include <libsolidity/ast/ASTForward.h> | ||||
| #include <libsolidity/ast/CallGraph.h> | ||||
| 
 | ||||
| #include <liblangutil/CharStreamProvider.h> | ||||
| #include <liblangutil/DebugInfoSelection.h> | ||||
| #include <liblangutil/EVMVersion.h> | ||||
| 
 | ||||
| #include <json/json.h> | ||||
| 
 | ||||
| #include <string> | ||||
| 
 | ||||
| namespace solidity::frontend::experimental | ||||
| { | ||||
| 
 | ||||
| class SourceUnit; | ||||
| 
 | ||||
| class IRGenerator | ||||
| { | ||||
| public: | ||||
| 	IRGenerator( | ||||
| 		langutil::EVMVersion _evmVersion, | ||||
| 		std::optional<uint8_t> _eofVersion, | ||||
| 		RevertStrings /*_revertStrings*/, | ||||
| 		std::map<std::string, unsigned> /*_sourceIndices*/, | ||||
| 		langutil::DebugInfoSelection const& _debugInfoSelection, | ||||
| 		langutil::CharStreamProvider const* _soliditySourceProvider | ||||
| 	): | ||||
| 		m_evmVersion(_evmVersion), | ||||
| 		m_eofVersion(_eofVersion), | ||||
| 		m_debugInfoSelection(_debugInfoSelection), | ||||
| 		m_soliditySourceProvider(_soliditySourceProvider) | ||||
| 	{} | ||||
| 
 | ||||
| 	std::string run( | ||||
| 		ContractDefinition const& _contract, | ||||
| 		bytes const& _cborMetadata, | ||||
| 		std::map<ContractDefinition const*, std::string_view const> const& _otherYulSources | ||||
| 	) const; | ||||
| 
 | ||||
| 	std::string generate(ContractDefinition const& _contract) const; | ||||
| 	std::string generate(FunctionDefinition const& _function) const; | ||||
| 	std::string generate(InlineAssembly const& _assembly) const; | ||||
| private: | ||||
| 	langutil::EVMVersion const m_evmVersion; | ||||
| 	std::optional<uint8_t> const m_eofVersion; | ||||
| 	OptimiserSettings const m_optimiserSettings; | ||||
| 	langutil::DebugInfoSelection m_debugInfoSelection = {}; | ||||
| 	langutil::CharStreamProvider const* m_soliditySourceProvider = nullptr; | ||||
| }; | ||||
| 
 | ||||
| } | ||||
| @ -57,6 +57,7 @@ | ||||
| #include <libsolidity/parsing/Parser.h> | ||||
| 
 | ||||
| #include <libsolidity/experimental/analysis/Analysis.h> | ||||
| #include <libsolidity/experimental/codegen/IRGenerator.h> | ||||
| 
 | ||||
| #include <libsolidity/codegen/ir/Common.h> | ||||
| #include <libsolidity/codegen/ir/IRGenerator.h> | ||||
| @ -310,6 +311,7 @@ void CompilerStack::reset(bool _keepSettings) | ||||
| { | ||||
| 	m_stackState = Empty; | ||||
| 	m_sources.clear(); | ||||
| 	m_maxAstId.reset(); | ||||
| 	m_smtlib2Responses.clear(); | ||||
| 	m_unhandledSMTLib2Queries.clear(); | ||||
| 	if (!_keepSettings) | ||||
| @ -410,6 +412,10 @@ bool CompilerStack::parse() | ||||
| 
 | ||||
| 	m_stackState = (m_stopAfter <= Parsed ? Parsed : ParsedAndImported); | ||||
| 	storeContractDefinitions(); | ||||
| 
 | ||||
| 	solAssert(!m_maxAstId.has_value()); | ||||
| 	m_maxAstId = parser.maxID(); | ||||
| 
 | ||||
| 	return true; | ||||
| } | ||||
| 
 | ||||
| @ -461,7 +467,7 @@ bool CompilerStack::analyze() | ||||
| 
 | ||||
| 		m_globalContext = std::make_shared<GlobalContext>(); | ||||
| 		// We need to keep the same resolver during the whole process.
 | ||||
| 		NameAndTypeResolver resolver(*m_globalContext, m_evmVersion, m_errorReporter); | ||||
| 		NameAndTypeResolver resolver(*m_globalContext, m_evmVersion, m_errorReporter, experimentalSolidity); | ||||
| 		for (Source const* source: m_sourceOrder) | ||||
| 			if (source->ast && !resolver.registerDeclarations(*source->ast)) | ||||
| 				return false; | ||||
| @ -475,10 +481,12 @@ bool CompilerStack::analyze() | ||||
| 
 | ||||
| 		resolver.warnHomonymDeclarations(); | ||||
| 
 | ||||
| 		{ | ||||
| 			DocStringTagParser docStringTagParser(m_errorReporter); | ||||
| 			for (Source const* source: m_sourceOrder) | ||||
| 				if (source->ast && !docStringTagParser.parseDocStrings(*source->ast)) | ||||
| 					noErrors = false; | ||||
|  		} | ||||
| 
 | ||||
| 		// Requires DocStringTagParser
 | ||||
| 		for (Source const* source: m_sourceOrder) | ||||
| @ -655,7 +663,8 @@ bool CompilerStack::analyzeLegacy(bool _noErrorsSoFar) | ||||
| bool CompilerStack::analyzeExperimental() | ||||
| { | ||||
| 	solAssert(!m_experimentalAnalysis); | ||||
| 	m_experimentalAnalysis = std::make_unique<experimental::Analysis>(m_errorReporter); | ||||
| 	solAssert(m_maxAstId && *m_maxAstId >= 0); | ||||
| 	m_experimentalAnalysis = std::make_unique<experimental::Analysis>(m_errorReporter, static_cast<std::uint64_t>(*m_maxAstId)); | ||||
| 	std::vector<std::shared_ptr<SourceUnit const>> sourceAsts; | ||||
| 	for (Source const* source: m_sourceOrder) | ||||
| 		if (source->ast) | ||||
| @ -1492,6 +1501,25 @@ void CompilerStack::generateIR(ContractDefinition const& _contract) | ||||
| 	for (auto const& pair: m_contracts) | ||||
| 		otherYulSources.emplace(pair.second.contract, pair.second.yulIR); | ||||
| 
 | ||||
| 	if (m_experimentalAnalysis) | ||||
| 	{ | ||||
| 		experimental::IRGenerator generator( | ||||
| 			m_evmVersion, | ||||
| 			m_eofVersion, | ||||
| 			m_revertStrings, | ||||
| 			sourceIndices(), | ||||
| 			m_debugInfoSelection, | ||||
| 			this, | ||||
| 			*m_experimentalAnalysis | ||||
| 		); | ||||
| 		compiledContract.yulIR = generator.run( | ||||
| 			_contract, | ||||
| 			{}, // TODO: createCBORMetadata(compiledContract, /* _forIR */ true),
 | ||||
| 			otherYulSources | ||||
| 		); | ||||
| 	} | ||||
| 	else | ||||
| 	{ | ||||
| 		IRGenerator generator( | ||||
| 			m_evmVersion, | ||||
| 			m_eofVersion, | ||||
| @ -1505,6 +1533,7 @@ void CompilerStack::generateIR(ContractDefinition const& _contract) | ||||
| 			createCBORMetadata(compiledContract, /* _forIR */ true), | ||||
| 			otherYulSources | ||||
| 		); | ||||
| 	} | ||||
| 
 | ||||
| 	yul::YulStack stack( | ||||
| 		m_evmVersion, | ||||
|  | ||||
| @ -225,6 +225,13 @@ public: | ||||
| 	/// @returns false on error.
 | ||||
| 	bool analyze(); | ||||
| 
 | ||||
| 	/// Perform the analysis steps of legacy language mode.
 | ||||
| 	/// @returns false on error.
 | ||||
| 	bool analyzeLegacy(bool _noErrorsSoFar); | ||||
| 	/// Perform the analysis steps of experimental language mode.
 | ||||
| 	/// @returns false on error.
 | ||||
| 	bool analyzeExperimental(); | ||||
| 
 | ||||
| 	/// Parses and analyzes all source units that were added
 | ||||
| 	/// @returns false on error.
 | ||||
| 	bool parseAndAnalyze(State _stopAfter = State::CompilationSuccessful); | ||||
|  | ||||
		Loading…
	
		Reference in New Issue
	
	Block a user