mirror of
				https://github.com/ethereum/solidity
				synced 2023-10-03 13:03:40 +00:00 
			
		
		
		
	
		
			
				
	
	
		
			197 lines
		
	
	
		
			6.6 KiB
		
	
	
	
		
			C++
		
	
	
	
	
	
			
		
		
	
	
			197 lines
		
	
	
		
			6.6 KiB
		
	
	
	
		
			C++
		
	
	
	
	
	
/*
 | 
						|
	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/>.
 | 
						|
*/
 | 
						|
/**
 | 
						|
 * @author Christian <c@ethdev.com>
 | 
						|
 * @date 2014
 | 
						|
 * JSON interface for the solidity compiler to be used from Javascript.
 | 
						|
 */
 | 
						|
 | 
						|
#include <string>
 | 
						|
#include <iostream>
 | 
						|
#include <json/json.h>
 | 
						|
#include <libdevcore/Common.h>
 | 
						|
#include <libdevcore/CommonData.h>
 | 
						|
#include <libdevcore/CommonIO.h>
 | 
						|
#include <libevmcore/Instruction.h>
 | 
						|
#include <libevmcore/Params.h>
 | 
						|
#include <libsolidity/Scanner.h>
 | 
						|
#include <libsolidity/Parser.h>
 | 
						|
#include <libsolidity/ASTPrinter.h>
 | 
						|
#include <libsolidity/NameAndTypeResolver.h>
 | 
						|
#include <libsolidity/Exceptions.h>
 | 
						|
#include <libsolidity/CompilerStack.h>
 | 
						|
#include <libsolidity/SourceReferenceFormatter.h>
 | 
						|
#include <libsolidity/ASTJsonConverter.h>
 | 
						|
 | 
						|
using namespace std;
 | 
						|
using namespace dev;
 | 
						|
using namespace solidity;
 | 
						|
 | 
						|
string formatError(Exception const& _exception, string const& _name, CompilerStack const& _compiler)
 | 
						|
{
 | 
						|
	ostringstream errorOutput;
 | 
						|
	SourceReferenceFormatter::printExceptionInformation(errorOutput, _exception, _name, _compiler);
 | 
						|
 | 
						|
	Json::Value output(Json::objectValue);
 | 
						|
	output["error"] = errorOutput.str();
 | 
						|
	return Json::FastWriter().write(output);
 | 
						|
}
 | 
						|
 | 
						|
Json::Value functionHashes(ContractDefinition const& _contract)
 | 
						|
{
 | 
						|
	Json::Value functionHashes(Json::objectValue);
 | 
						|
	for (auto const& it: _contract.interfaceFunctions())
 | 
						|
		functionHashes[it.second->externalSignature()] = toHex(it.first.ref());
 | 
						|
	return functionHashes;
 | 
						|
}
 | 
						|
 | 
						|
Json::Value gasToJson(GasEstimator::GasConsumption const& _gas)
 | 
						|
{
 | 
						|
	if (_gas.isInfinite || _gas.value > std::numeric_limits<Json::LargestUInt>::max())
 | 
						|
		return Json::Value(Json::nullValue);
 | 
						|
	else
 | 
						|
		return Json::Value(Json::LargestUInt(_gas.value));
 | 
						|
}
 | 
						|
 | 
						|
Json::Value estimateGas(CompilerStack const& _compiler, string const& _contract)
 | 
						|
{
 | 
						|
	Json::Value gasEstimates(Json::objectValue);
 | 
						|
	using Gas = GasEstimator::GasConsumption;
 | 
						|
	if (!_compiler.assemblyItems(_contract) && !_compiler.runtimeAssemblyItems(_contract))
 | 
						|
		return gasEstimates;
 | 
						|
	if (eth::AssemblyItems const* items = _compiler.assemblyItems(_contract))
 | 
						|
	{
 | 
						|
		Gas gas = GasEstimator::functionalEstimation(*items);
 | 
						|
		u256 bytecodeSize(_compiler.runtimeObject(_contract).bytecode.size());
 | 
						|
		Json::Value creationGas(Json::arrayValue);
 | 
						|
		creationGas[0] = gasToJson(gas);
 | 
						|
		creationGas[1] = gasToJson(bytecodeSize * eth::c_createDataGas);
 | 
						|
		gasEstimates["creation"] = creationGas;
 | 
						|
	}
 | 
						|
	if (eth::AssemblyItems const* items = _compiler.runtimeAssemblyItems(_contract))
 | 
						|
	{
 | 
						|
		ContractDefinition const& contract = _compiler.contractDefinition(_contract);
 | 
						|
		Json::Value externalFunctions(Json::objectValue);
 | 
						|
		for (auto it: contract.interfaceFunctions())
 | 
						|
		{
 | 
						|
			string sig = it.second->externalSignature();
 | 
						|
			externalFunctions[sig] = gasToJson(GasEstimator::functionalEstimation(*items, sig));
 | 
						|
		}
 | 
						|
		if (contract.fallbackFunction())
 | 
						|
			externalFunctions[""] = gasToJson(GasEstimator::functionalEstimation(*items, "INVALID"));
 | 
						|
		gasEstimates["external"] = externalFunctions;
 | 
						|
		Json::Value internalFunctions(Json::objectValue);
 | 
						|
		for (auto const& it: contract.definedFunctions())
 | 
						|
		{
 | 
						|
			if (it->isPartOfExternalInterface() || it->isConstructor())
 | 
						|
				continue;
 | 
						|
			size_t entry = _compiler.functionEntryPoint(_contract, *it);
 | 
						|
			GasEstimator::GasConsumption gas = GasEstimator::GasConsumption::infinite();
 | 
						|
			if (entry > 0)
 | 
						|
				gas = GasEstimator::functionalEstimation(*items, entry, *it);
 | 
						|
			FunctionType type(*it);
 | 
						|
			string sig = it->name() + "(";
 | 
						|
			auto end = type.parameterTypes().end();
 | 
						|
			for (auto it = type.parameterTypes().begin(); it != end; ++it)
 | 
						|
				sig += (*it)->toString() + (it + 1 == end ? "" : ",");
 | 
						|
			sig += ")";
 | 
						|
			internalFunctions[sig] = gasToJson(gas);
 | 
						|
		}
 | 
						|
		gasEstimates["internal"] = internalFunctions;
 | 
						|
	}
 | 
						|
	return gasEstimates;
 | 
						|
}
 | 
						|
 | 
						|
string compile(string _input, bool _optimize)
 | 
						|
{
 | 
						|
	StringMap sources;
 | 
						|
	sources[""] = _input;
 | 
						|
 | 
						|
	Json::Value output(Json::objectValue);
 | 
						|
	CompilerStack compiler;
 | 
						|
	try
 | 
						|
	{
 | 
						|
		compiler.compile(_input, _optimize);
 | 
						|
	}
 | 
						|
	catch (ParserError const& exception)
 | 
						|
	{
 | 
						|
		return formatError(exception, "Parser error", compiler);
 | 
						|
	}
 | 
						|
	catch (DeclarationError const& exception)
 | 
						|
	{
 | 
						|
		return formatError(exception, "Declaration error", compiler);
 | 
						|
	}
 | 
						|
	catch (TypeError const& exception)
 | 
						|
	{
 | 
						|
		return formatError(exception, "Type error", compiler);
 | 
						|
	}
 | 
						|
	catch (CompilerError const& exception)
 | 
						|
	{
 | 
						|
		return formatError(exception, "Compiler error", compiler);
 | 
						|
	}
 | 
						|
	catch (InternalCompilerError const& exception)
 | 
						|
	{
 | 
						|
		return formatError(exception, "Internal compiler error", compiler);
 | 
						|
	}
 | 
						|
	catch (DocstringParsingError const& exception)
 | 
						|
	{
 | 
						|
		return formatError(exception, "Documentation parsing error", compiler);
 | 
						|
	}
 | 
						|
	catch (Exception const& exception)
 | 
						|
	{
 | 
						|
		output["error"] = "Exception during compilation: " + boost::diagnostic_information(exception);
 | 
						|
		return Json::FastWriter().write(output);
 | 
						|
	}
 | 
						|
	catch (...)
 | 
						|
	{
 | 
						|
		output["error"] = "Unknown exception during compilation.";
 | 
						|
		return Json::FastWriter().write(output);
 | 
						|
	}
 | 
						|
 | 
						|
	output["contracts"] = Json::Value(Json::objectValue);
 | 
						|
	for (string const& contractName: compiler.contractNames())
 | 
						|
	{
 | 
						|
		Json::Value contractData(Json::objectValue);
 | 
						|
		contractData["solidity_interface"] = compiler.solidityInterface(contractName);
 | 
						|
		contractData["interface"] = compiler.interface(contractName);
 | 
						|
		contractData["bytecode"] = compiler.object(contractName).toHex();
 | 
						|
		contractData["opcodes"] = eth::disassemble(compiler.object(contractName).bytecode);
 | 
						|
		contractData["functionHashes"] = functionHashes(compiler.contractDefinition(contractName));
 | 
						|
		contractData["gasEstimates"] = estimateGas(compiler, contractName);
 | 
						|
		ostringstream unused;
 | 
						|
		contractData["assembly"] = compiler.streamAssembly(unused, contractName, sources, true);
 | 
						|
		output["contracts"][contractName] = contractData;
 | 
						|
	}
 | 
						|
 | 
						|
	output["sources"] = Json::Value(Json::objectValue);
 | 
						|
	output["sources"][""] = Json::Value(Json::objectValue);
 | 
						|
	output["sources"][""]["AST"] = ASTJsonConverter(compiler.ast("")).json();
 | 
						|
 | 
						|
	return Json::FastWriter().write(output);
 | 
						|
}
 | 
						|
 | 
						|
static string outputBuffer;
 | 
						|
 | 
						|
extern "C"
 | 
						|
{
 | 
						|
extern char const* compileJSON(char const* _input, bool _optimize)
 | 
						|
{
 | 
						|
	outputBuffer = compile(_input, _optimize);
 | 
						|
	return outputBuffer.c_str();
 | 
						|
}
 | 
						|
}
 |