mirror of
				https://github.com/ethereum/solidity
				synced 2023-10-03 13:03:40 +00:00 
			
		
		
		
	
		
			
				
	
	
		
			340 lines
		
	
	
		
			11 KiB
		
	
	
	
		
			C++
		
	
	
	
	
	
			
		
		
	
	
			340 lines
		
	
	
		
			11 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 <functional>
 | |
| #include <iostream>
 | |
| #include <json/json.h>
 | |
| #include <libdevcore/Common.h>
 | |
| #include <libdevcore/CommonData.h>
 | |
| #include <libdevcore/CommonIO.h>
 | |
| #include <libdevcore/JSON.h>
 | |
| #include <libevmasm/Instruction.h>
 | |
| #include <libevmasm/GasMeter.h>
 | |
| #include <libsolidity/parsing/Scanner.h>
 | |
| #include <libsolidity/parsing/Parser.h>
 | |
| #include <libsolidity/ast/ASTPrinter.h>
 | |
| #include <libsolidity/analysis/NameAndTypeResolver.h>
 | |
| #include <libsolidity/interface/Exceptions.h>
 | |
| #include <libsolidity/interface/CompilerStack.h>
 | |
| #include <libsolidity/interface/SourceReferenceFormatter.h>
 | |
| #include <libsolidity/ast/ASTJsonConverter.h>
 | |
| #include <libsolidity/interface/Version.h>
 | |
| 
 | |
| using namespace std;
 | |
| using namespace dev;
 | |
| using namespace solidity;
 | |
| 
 | |
| extern "C" {
 | |
| /// Callback used to retrieve additional source files. "Returns" two pointers that should be
 | |
| /// heap-allocated and are free'd by the caller.
 | |
| typedef void (*CStyleReadFileCallback)(char const* _path, char** o_contents, char** o_error);
 | |
| }
 | |
| 
 | |
| string formatError(
 | |
| 	Exception const& _exception,
 | |
| 	string const& _name,
 | |
| 	function<Scanner const&(string const&)> const& _scannerFromSourceName
 | |
| )
 | |
| {
 | |
| 	ostringstream errorOutput;
 | |
| 	SourceReferenceFormatter::printExceptionInformation(errorOutput, _exception, _name, _scannerFromSourceName);
 | |
| 	return errorOutput.str();
 | |
| }
 | |
| 
 | |
| 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::GasCosts::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 paramTypes = type.parameterTypes();
 | |
| 			for (auto it = paramTypes.begin(); it != paramTypes.end(); ++it)
 | |
| 				sig += (*it)->toString() + (it + 1 == paramTypes.end() ? "" : ",");
 | |
| 			sig += ")";
 | |
| 			internalFunctions[sig] = gasToJson(gas);
 | |
| 		}
 | |
| 		gasEstimates["internal"] = internalFunctions;
 | |
| 	}
 | |
| 	return gasEstimates;
 | |
| }
 | |
| 
 | |
| string compile(StringMap const& _sources, bool _optimize, CStyleReadFileCallback _readCallback)
 | |
| {
 | |
| 	Json::Value output(Json::objectValue);
 | |
| 	Json::Value errors(Json::arrayValue);
 | |
| 	CompilerStack::ReadFileCallback readCallback;
 | |
| 	if (_readCallback)
 | |
| 	{
 | |
| 		readCallback = [=](string const& _path)
 | |
| 		{
 | |
| 			char* contents_c = nullptr;
 | |
| 			char* error_c = nullptr;
 | |
| 			_readCallback(_path.c_str(), &contents_c, &error_c);
 | |
| 			CompilerStack::ReadFileResult result;
 | |
| 			result.success = true;
 | |
| 			if (!contents_c && !error_c)
 | |
| 			{
 | |
| 				result.success = false;
 | |
| 				result.contentsOrErrorMesage = "File not found.";
 | |
| 			}
 | |
| 			if (contents_c)
 | |
| 			{
 | |
| 				result.success = true;
 | |
| 				result.contentsOrErrorMesage = string(contents_c);
 | |
| 				free(contents_c);
 | |
| 			}
 | |
| 			if (error_c)
 | |
| 			{
 | |
| 				result.success = false;
 | |
| 				result.contentsOrErrorMesage = string(error_c);
 | |
| 				free(error_c);
 | |
| 			}
 | |
| 			return result;
 | |
| 		};
 | |
| 	}
 | |
| 	CompilerStack compiler(readCallback);
 | |
| 	auto scannerFromSourceName = [&](string const& _sourceName) -> solidity::Scanner const& { return compiler.scanner(_sourceName); };
 | |
| 	bool success = false;
 | |
| 	try
 | |
| 	{
 | |
| 		compiler.addSources(_sources);
 | |
| 		bool succ = compiler.compile(_optimize);
 | |
| 		for (auto const& error: compiler.errors())
 | |
| 		{
 | |
| 			auto err = dynamic_pointer_cast<Error const>(error);
 | |
| 			errors.append(formatError(
 | |
| 				*error,
 | |
| 				(err->type() == Error::Type::Warning) ? "Warning" : "Error",
 | |
| 				scannerFromSourceName
 | |
| 			));
 | |
| 		}
 | |
| 		success = succ; // keep success false on exception
 | |
| 	}
 | |
| 	catch (Error const& error)
 | |
| 	{
 | |
| 		errors.append(formatError(error, error.typeName(), scannerFromSourceName));
 | |
| 	}
 | |
| 	catch (CompilerError const& exception)
 | |
| 	{
 | |
| 		errors.append(formatError(exception, "Compiler error", scannerFromSourceName));
 | |
| 	}
 | |
| 	catch (InternalCompilerError const& exception)
 | |
| 	{
 | |
| 		errors.append(formatError(exception, "Internal compiler error", scannerFromSourceName));
 | |
| 	}
 | |
| 	catch (UnimplementedFeatureError const& exception)
 | |
| 	{
 | |
| 		errors.append(formatError(exception, "Unimplemented feature", scannerFromSourceName));
 | |
| 	}
 | |
| 	catch (Exception const& exception)
 | |
| 	{
 | |
| 		errors.append("Exception during compilation: " + boost::diagnostic_information(exception));
 | |
| 	}
 | |
| 	catch (...)
 | |
| 	{
 | |
| 		errors.append("Unknown exception during compilation.");
 | |
| 	}
 | |
| 
 | |
| 	if (errors.size() > 0)
 | |
| 		output["errors"] = errors;
 | |
| 
 | |
| 	if (success)
 | |
| 	{
 | |
| 		try
 | |
| 		{
 | |
| 			output["contracts"] = Json::Value(Json::objectValue);
 | |
| 			for (string const& contractName: compiler.contractNames())
 | |
| 			{
 | |
| 				Json::Value contractData(Json::objectValue);
 | |
| 				contractData["interface"] = dev::jsonCompactPrint(compiler.interface(contractName));
 | |
| 				contractData["bytecode"] = compiler.object(contractName).toHex();
 | |
| 				contractData["runtimeBytecode"] = compiler.runtimeObject(contractName).toHex();
 | |
| 				contractData["opcodes"] = solidity::disassemble(compiler.object(contractName).bytecode);
 | |
| 				contractData["functionHashes"] = functionHashes(compiler.contractDefinition(contractName));
 | |
| 				contractData["gasEstimates"] = estimateGas(compiler, contractName);
 | |
| 				auto sourceMap = compiler.sourceMapping(contractName);
 | |
| 				contractData["srcmap"] = sourceMap ? *sourceMap : "";
 | |
| 				auto runtimeSourceMap = compiler.runtimeSourceMapping(contractName);
 | |
| 				contractData["srcmapRuntime"] = runtimeSourceMap ? *runtimeSourceMap : "";
 | |
| 				ostringstream unused;
 | |
| 				contractData["assembly"] = compiler.streamAssembly(unused, contractName, _sources, true);
 | |
| 				output["contracts"][contractName] = contractData;
 | |
| 			}
 | |
| 		}
 | |
| 		catch (...)
 | |
| 		{
 | |
| 			output["errors"].append("Unknown exception while generating contract data output.");
 | |
| 		}
 | |
| 
 | |
| 		try
 | |
| 		{
 | |
| 			// Do not taint the internal error list
 | |
| 			ErrorList formalErrors;
 | |
| 			if (compiler.prepareFormalAnalysis(&formalErrors))
 | |
| 				output["formal"]["why3"] = compiler.formalTranslation();
 | |
| 			if (!formalErrors.empty())
 | |
| 			{
 | |
| 				Json::Value errors(Json::arrayValue);
 | |
| 				for (auto const& error: formalErrors)
 | |
| 					errors.append(formatError(
 | |
| 						*error,
 | |
| 						(error->type() == Error::Type::Warning) ? "Warning" : "Error",
 | |
| 						scannerFromSourceName
 | |
| 					));
 | |
| 				output["formal"]["errors"] = errors;
 | |
| 			}
 | |
| 		}
 | |
| 		catch (...)
 | |
| 		{
 | |
| 			output["errors"].append("Unknown exception while generating formal method output.");
 | |
| 		}
 | |
| 
 | |
| 		try
 | |
| 		{
 | |
| 			// Indices into this array are used to abbreviate source names in source locations.
 | |
| 			output["sourceList"] = Json::Value(Json::arrayValue);
 | |
| 			for (auto const& source: compiler.sourceNames())
 | |
| 				output["sourceList"].append(source);
 | |
| 			output["sources"] = Json::Value(Json::objectValue);
 | |
| 			for (auto const& source: compiler.sourceNames())
 | |
| 				output["sources"][source]["AST"] = ASTJsonConverter(compiler.ast(source), compiler.sourceIndices()).json();
 | |
| 		}
 | |
| 		catch (...)
 | |
| 		{
 | |
| 			output["errors"].append("Unknown exception while generating source name output.");
 | |
| 		}
 | |
| 	}
 | |
| 
 | |
| 	try
 | |
| 	{
 | |
| 		return dev::jsonCompactPrint(output);
 | |
| 	}
 | |
| 	catch (...)
 | |
| 	{
 | |
| 		return "{\"errors\":[\"Unknown error while generating JSON.\"]}";
 | |
| 	}
 | |
| }
 | |
| 
 | |
| string compileMulti(string const& _input, bool _optimize, CStyleReadFileCallback _readCallback = nullptr)
 | |
| {
 | |
| 	Json::Reader reader;
 | |
| 	Json::Value input;
 | |
| 	if (!reader.parse(_input, input, false))
 | |
| 	{
 | |
| 		Json::Value errors(Json::arrayValue);
 | |
| 		errors.append("Error parsing input JSON: " + reader.getFormattedErrorMessages());
 | |
| 		Json::Value output(Json::objectValue);
 | |
| 		output["errors"] = errors;
 | |
| 		return dev::jsonCompactPrint(output);
 | |
| 	}
 | |
| 	else
 | |
| 	{
 | |
| 		StringMap sources;
 | |
| 		Json::Value jsonSources = input["sources"];
 | |
| 		if (jsonSources.isObject())
 | |
| 			for (auto const& sourceName: jsonSources.getMemberNames())
 | |
| 				sources[sourceName] = jsonSources[sourceName].asString();
 | |
| 		return compile(sources, _optimize, _readCallback);
 | |
| 	}
 | |
| }
 | |
| 
 | |
| string compileSingle(string const& _input, bool _optimize)
 | |
| {
 | |
| 	StringMap sources;
 | |
| 	sources[""] = _input;
 | |
| 	return compile(sources, _optimize, nullptr);
 | |
| }
 | |
| 
 | |
| static string s_outputBuffer;
 | |
| 
 | |
| extern "C"
 | |
| {
 | |
| extern char const* version()
 | |
| {
 | |
| 	return VersionString.c_str();
 | |
| }
 | |
| extern char const* compileJSON(char const* _input, bool _optimize)
 | |
| {
 | |
| 	s_outputBuffer = compileSingle(_input, _optimize);
 | |
| 	return s_outputBuffer.c_str();
 | |
| }
 | |
| extern char const* compileJSONMulti(char const* _input, bool _optimize)
 | |
| {
 | |
| 	s_outputBuffer = compileMulti(_input, _optimize);
 | |
| 	return s_outputBuffer.c_str();
 | |
| }
 | |
| extern char const* compileJSONCallback(char const* _input, bool _optimize, CStyleReadFileCallback _readCallback)
 | |
| {
 | |
| 	s_outputBuffer = compileMulti(_input, _optimize, _readCallback);
 | |
| 	return s_outputBuffer.c_str();
 | |
| }
 | |
| }
 |