mirror of
				https://github.com/ethereum/solidity
				synced 2023-10-03 13:03:40 +00:00 
			
		
		
		
	
		
			
				
	
	
		
			1242 lines
		
	
	
		
			38 KiB
		
	
	
	
		
			C++
		
	
	
	
	
	
			
		
		
	
	
			1242 lines
		
	
	
		
			38 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/>.
 | 
						|
*/
 | 
						|
/**
 | 
						|
 * @author Lefteris <lefteris@ethdev.com>
 | 
						|
 * @author Gav Wood <g@ethdev.com>
 | 
						|
 * @date 2014
 | 
						|
 * Solidity command line interface.
 | 
						|
 */
 | 
						|
#include "CommandLineInterface.h"
 | 
						|
 | 
						|
#include "solidity/BuildInfo.h"
 | 
						|
#include "license.h"
 | 
						|
 | 
						|
#include <libsolidity/interface/Version.h>
 | 
						|
#include <libsolidity/parsing/Scanner.h>
 | 
						|
#include <libsolidity/parsing/Parser.h>
 | 
						|
#include <libsolidity/ast/ASTPrinter.h>
 | 
						|
#include <libsolidity/ast/ASTJsonConverter.h>
 | 
						|
#include <libsolidity/analysis/NameAndTypeResolver.h>
 | 
						|
#include <libsolidity/interface/Exceptions.h>
 | 
						|
#include <libsolidity/interface/CompilerStack.h>
 | 
						|
#include <libsolidity/interface/StandardCompiler.h>
 | 
						|
#include <libsolidity/interface/SourceReferenceFormatter.h>
 | 
						|
#include <libsolidity/interface/GasEstimator.h>
 | 
						|
#include <libsolidity/interface/AssemblyStack.h>
 | 
						|
 | 
						|
#include <libevmasm/Instruction.h>
 | 
						|
#include <libevmasm/GasMeter.h>
 | 
						|
 | 
						|
#include <libdevcore/Common.h>
 | 
						|
#include <libdevcore/CommonData.h>
 | 
						|
#include <libdevcore/CommonIO.h>
 | 
						|
#include <libdevcore/JSON.h>
 | 
						|
 | 
						|
#include <boost/filesystem.hpp>
 | 
						|
#include <boost/filesystem/operations.hpp>
 | 
						|
#include <boost/algorithm/string.hpp>
 | 
						|
 | 
						|
#ifdef _WIN32 // windows
 | 
						|
	#include <io.h>
 | 
						|
	#define isatty _isatty
 | 
						|
	#define fileno _fileno
 | 
						|
#else // unix
 | 
						|
	#include <unistd.h>
 | 
						|
#endif
 | 
						|
#include <string>
 | 
						|
#include <iostream>
 | 
						|
#include <fstream>
 | 
						|
 | 
						|
using namespace std;
 | 
						|
namespace po = boost::program_options;
 | 
						|
 | 
						|
namespace dev
 | 
						|
{
 | 
						|
namespace solidity
 | 
						|
{
 | 
						|
 | 
						|
static string const g_stdinFileNameStr = "<stdin>";
 | 
						|
static string const g_strAbi = "abi";
 | 
						|
static string const g_strAllowPaths = "allow-paths";
 | 
						|
static string const g_strAsm = "asm";
 | 
						|
static string const g_strAsmJson = "asm-json";
 | 
						|
static string const g_strAssemble = "assemble";
 | 
						|
static string const g_strAst = "ast";
 | 
						|
static string const g_strAstJson = "ast-json";
 | 
						|
static string const g_strAstCompactJson = "ast-compact-json";
 | 
						|
static string const g_strBinary = "bin";
 | 
						|
static string const g_strBinaryRuntime = "bin-runtime";
 | 
						|
static string const g_strCloneBinary = "clone-bin";
 | 
						|
static string const g_strCombinedJson = "combined-json";
 | 
						|
static string const g_strCompactJSON = "compact-format";
 | 
						|
static string const g_strContracts = "contracts";
 | 
						|
static string const g_strEVM = "evm";
 | 
						|
static string const g_strEVM15 = "evm15";
 | 
						|
static string const g_strEVMVersion = "evm-version";
 | 
						|
static string const g_streWasm = "ewasm";
 | 
						|
static string const g_strFormal = "formal";
 | 
						|
static string const g_strGas = "gas";
 | 
						|
static string const g_strHelp = "help";
 | 
						|
static string const g_strInputFile = "input-file";
 | 
						|
static string const g_strInterface = "interface";
 | 
						|
static string const g_strJulia = "julia";
 | 
						|
static string const g_strLicense = "license";
 | 
						|
static string const g_strLibraries = "libraries";
 | 
						|
static string const g_strLink = "link";
 | 
						|
static string const g_strMachine = "machine";
 | 
						|
static string const g_strMetadata = "metadata";
 | 
						|
static string const g_strMetadataLiteral = "metadata-literal";
 | 
						|
static string const g_strNatspecDev = "devdoc";
 | 
						|
static string const g_strNatspecUser = "userdoc";
 | 
						|
static string const g_strOpcodes = "opcodes";
 | 
						|
static string const g_strOptimize = "optimize";
 | 
						|
static string const g_strOptimizeRuns = "optimize-runs";
 | 
						|
static string const g_strOutputDir = "output-dir";
 | 
						|
static string const g_strOverwrite = "overwrite";
 | 
						|
static string const g_strSignatureHashes = "hashes";
 | 
						|
static string const g_strSources = "sources";
 | 
						|
static string const g_strSourceList = "sourceList";
 | 
						|
static string const g_strSrcMap = "srcmap";
 | 
						|
static string const g_strSrcMapRuntime = "srcmap-runtime";
 | 
						|
static string const g_strStandardJSON = "standard-json";
 | 
						|
static string const g_strStrictAssembly = "strict-assembly";
 | 
						|
static string const g_strPrettyJson = "pretty-json";
 | 
						|
static string const g_strVersion = "version";
 | 
						|
static string const g_strIgnoreMissingFiles = "ignore-missing";
 | 
						|
 | 
						|
static string const g_argAbi = g_strAbi;
 | 
						|
static string const g_argPrettyJson = g_strPrettyJson;
 | 
						|
static string const g_argAllowPaths = g_strAllowPaths;
 | 
						|
static string const g_argAsm = g_strAsm;
 | 
						|
static string const g_argAsmJson = g_strAsmJson;
 | 
						|
static string const g_argAssemble = g_strAssemble;
 | 
						|
static string const g_argAst = g_strAst;
 | 
						|
static string const g_argAstCompactJson = g_strAstCompactJson;
 | 
						|
static string const g_argAstJson = g_strAstJson;
 | 
						|
static string const g_argBinary = g_strBinary;
 | 
						|
static string const g_argBinaryRuntime = g_strBinaryRuntime;
 | 
						|
static string const g_argCloneBinary = g_strCloneBinary;
 | 
						|
static string const g_argCombinedJson = g_strCombinedJson;
 | 
						|
static string const g_argCompactJSON = g_strCompactJSON;
 | 
						|
static string const g_argFormal = g_strFormal;
 | 
						|
static string const g_argGas = g_strGas;
 | 
						|
static string const g_argHelp = g_strHelp;
 | 
						|
static string const g_argInputFile = g_strInputFile;
 | 
						|
static string const g_argJulia = g_strJulia;
 | 
						|
static string const g_argLibraries = g_strLibraries;
 | 
						|
static string const g_argLink = g_strLink;
 | 
						|
static string const g_argMachine = g_strMachine;
 | 
						|
static string const g_argMetadata = g_strMetadata;
 | 
						|
static string const g_argMetadataLiteral = g_strMetadataLiteral;
 | 
						|
static string const g_argNatspecDev = g_strNatspecDev;
 | 
						|
static string const g_argNatspecUser = g_strNatspecUser;
 | 
						|
static string const g_argOpcodes = g_strOpcodes;
 | 
						|
static string const g_argOptimize = g_strOptimize;
 | 
						|
static string const g_argOptimizeRuns = g_strOptimizeRuns;
 | 
						|
static string const g_argOutputDir = g_strOutputDir;
 | 
						|
static string const g_argSignatureHashes = g_strSignatureHashes;
 | 
						|
static string const g_argStandardJSON = g_strStandardJSON;
 | 
						|
static string const g_argStrictAssembly = g_strStrictAssembly;
 | 
						|
static string const g_argVersion = g_strVersion;
 | 
						|
static string const g_stdinFileName = g_stdinFileNameStr;
 | 
						|
static string const g_argIgnoreMissingFiles = g_strIgnoreMissingFiles;
 | 
						|
 | 
						|
/// Possible arguments to for --combined-json
 | 
						|
static set<string> const g_combinedJsonArgs
 | 
						|
{
 | 
						|
	g_strAbi,
 | 
						|
	g_strAsm,
 | 
						|
	g_strAst,
 | 
						|
	g_strBinary,
 | 
						|
	g_strBinaryRuntime,
 | 
						|
	g_strCloneBinary,
 | 
						|
	g_strCompactJSON,
 | 
						|
	g_strInterface,
 | 
						|
	g_strMetadata,
 | 
						|
	g_strNatspecUser,
 | 
						|
	g_strNatspecDev,
 | 
						|
	g_strOpcodes,
 | 
						|
	g_strSignatureHashes,
 | 
						|
	g_strSrcMap,
 | 
						|
	g_strSrcMapRuntime
 | 
						|
};
 | 
						|
 | 
						|
/// Possible arguments to for --machine
 | 
						|
static set<string> const g_machineArgs
 | 
						|
{
 | 
						|
	g_strEVM,
 | 
						|
	g_strEVM15,
 | 
						|
	g_streWasm
 | 
						|
};
 | 
						|
 | 
						|
static void version()
 | 
						|
{
 | 
						|
	cout <<
 | 
						|
		"solc, the solidity compiler commandline interface" <<
 | 
						|
		endl <<
 | 
						|
		"Version: " <<
 | 
						|
		dev::solidity::VersionString <<
 | 
						|
		endl;
 | 
						|
	exit(0);
 | 
						|
}
 | 
						|
 | 
						|
static void license()
 | 
						|
{
 | 
						|
	cout << otherLicenses << endl;
 | 
						|
	// This is a static variable generated by cmake from LICENSE.txt
 | 
						|
	cout << licenseText << endl;
 | 
						|
	exit(0);
 | 
						|
}
 | 
						|
 | 
						|
static bool needsHumanTargetedStdout(po::variables_map const& _args)
 | 
						|
{
 | 
						|
	if (_args.count(g_argGas))
 | 
						|
		return true;
 | 
						|
	if (_args.count(g_argOutputDir))
 | 
						|
		return false;
 | 
						|
	for (string const& arg: {
 | 
						|
		g_argAbi,
 | 
						|
		g_argAsm,
 | 
						|
		g_argAsmJson,
 | 
						|
		g_argAstJson,
 | 
						|
		g_argBinary,
 | 
						|
		g_argBinaryRuntime,
 | 
						|
		g_argCloneBinary,
 | 
						|
		g_argFormal,
 | 
						|
		g_argMetadata,
 | 
						|
		g_argNatspecUser,
 | 
						|
		g_argNatspecDev,
 | 
						|
		g_argOpcodes,
 | 
						|
		g_argSignatureHashes
 | 
						|
	})
 | 
						|
		if (_args.count(arg))
 | 
						|
			return true;
 | 
						|
	return false;
 | 
						|
}
 | 
						|
 | 
						|
void CommandLineInterface::handleBinary(string const& _contract)
 | 
						|
{
 | 
						|
	if (m_args.count(g_argBinary))
 | 
						|
	{
 | 
						|
		if (m_args.count(g_argOutputDir))
 | 
						|
			createFile(m_compiler->filesystemFriendlyName(_contract) + ".bin", m_compiler->object(_contract).toHex());
 | 
						|
		else
 | 
						|
		{
 | 
						|
			cout << "Binary: " << endl;
 | 
						|
			cout << m_compiler->object(_contract).toHex() << endl;
 | 
						|
		}
 | 
						|
	}
 | 
						|
	if (m_args.count(g_argCloneBinary))
 | 
						|
	{
 | 
						|
		if (m_args.count(g_argOutputDir))
 | 
						|
			createFile(m_compiler->filesystemFriendlyName(_contract) + ".clone_bin", m_compiler->cloneObject(_contract).toHex());
 | 
						|
		else
 | 
						|
		{
 | 
						|
			cout << "Clone Binary: " << endl;
 | 
						|
			cout << m_compiler->cloneObject(_contract).toHex() << endl;
 | 
						|
		}
 | 
						|
	}
 | 
						|
	if (m_args.count(g_argBinaryRuntime))
 | 
						|
	{
 | 
						|
		if (m_args.count(g_argOutputDir))
 | 
						|
			createFile(m_compiler->filesystemFriendlyName(_contract) + ".bin-runtime", m_compiler->runtimeObject(_contract).toHex());
 | 
						|
		else
 | 
						|
		{
 | 
						|
			cout << "Binary of the runtime part: " << endl;
 | 
						|
			cout << m_compiler->runtimeObject(_contract).toHex() << endl;
 | 
						|
		}
 | 
						|
	}
 | 
						|
}
 | 
						|
 | 
						|
void CommandLineInterface::handleOpcode(string const& _contract)
 | 
						|
{
 | 
						|
	if (m_args.count(g_argOutputDir))
 | 
						|
		createFile(m_compiler->filesystemFriendlyName(_contract) + ".opcode", solidity::disassemble(m_compiler->object(_contract).bytecode));
 | 
						|
	else
 | 
						|
	{
 | 
						|
		cout << "Opcodes: " << endl;
 | 
						|
		cout << solidity::disassemble(m_compiler->object(_contract).bytecode);
 | 
						|
		cout << endl;
 | 
						|
	}
 | 
						|
}
 | 
						|
 | 
						|
void CommandLineInterface::handleBytecode(string const& _contract)
 | 
						|
{
 | 
						|
	if (m_args.count(g_argOpcodes))
 | 
						|
		handleOpcode(_contract);
 | 
						|
	if (m_args.count(g_argBinary) || m_args.count(g_argCloneBinary) || m_args.count(g_argBinaryRuntime))
 | 
						|
		handleBinary(_contract);
 | 
						|
}
 | 
						|
 | 
						|
void CommandLineInterface::handleSignatureHashes(string const& _contract)
 | 
						|
{
 | 
						|
	if (!m_args.count(g_argSignatureHashes))
 | 
						|
		return;
 | 
						|
 | 
						|
	Json::Value methodIdentifiers = m_compiler->methodIdentifiers(_contract);
 | 
						|
	string out;
 | 
						|
	for (auto const& name: methodIdentifiers.getMemberNames())
 | 
						|
		out += methodIdentifiers[name].asString() + ": " + name + "\n";
 | 
						|
 | 
						|
	if (m_args.count(g_argOutputDir))
 | 
						|
		createFile(m_compiler->filesystemFriendlyName(_contract) + ".signatures", out);
 | 
						|
	else
 | 
						|
		cout << "Function signatures: " << endl << out;
 | 
						|
}
 | 
						|
 | 
						|
void CommandLineInterface::handleMetadata(string const& _contract)
 | 
						|
{
 | 
						|
	if (!m_args.count(g_argMetadata))
 | 
						|
		return;
 | 
						|
 | 
						|
	string data = m_compiler->metadata(_contract);
 | 
						|
	if (m_args.count(g_argOutputDir))
 | 
						|
		createFile(m_compiler->filesystemFriendlyName(_contract) + "_meta.json", data);
 | 
						|
	else
 | 
						|
		cout << "Metadata: " << endl << data << endl;
 | 
						|
}
 | 
						|
 | 
						|
void CommandLineInterface::handleABI(string const& _contract)
 | 
						|
{
 | 
						|
	if (!m_args.count(g_argAbi))
 | 
						|
		return;
 | 
						|
 | 
						|
	string data = dev::jsonCompactPrint(m_compiler->contractABI(_contract));
 | 
						|
	if (m_args.count(g_argOutputDir))
 | 
						|
		createFile(m_compiler->filesystemFriendlyName(_contract) + ".abi", data);
 | 
						|
	else
 | 
						|
		cout << "Contract JSON ABI " << endl << data << endl;
 | 
						|
}
 | 
						|
 | 
						|
void CommandLineInterface::handleNatspec(bool _natspecDev, string const& _contract)
 | 
						|
{
 | 
						|
	std::string argName;
 | 
						|
	std::string suffix;
 | 
						|
	std::string title;
 | 
						|
 | 
						|
	if (_natspecDev)
 | 
						|
	{
 | 
						|
		argName = g_argNatspecDev;
 | 
						|
		suffix = ".docdev";
 | 
						|
		title = "Developer Documentation";
 | 
						|
	}
 | 
						|
	else
 | 
						|
	{
 | 
						|
		argName = g_argNatspecUser;
 | 
						|
		suffix = ".docuser";
 | 
						|
		title = "User Documentation";
 | 
						|
	}
 | 
						|
 | 
						|
	if (m_args.count(argName))
 | 
						|
	{
 | 
						|
		std::string output = dev::jsonPrettyPrint(
 | 
						|
			_natspecDev ?
 | 
						|
			m_compiler->natspecDev(_contract) :
 | 
						|
			m_compiler->natspecUser(_contract)
 | 
						|
		);
 | 
						|
 | 
						|
		if (m_args.count(g_argOutputDir))
 | 
						|
			createFile(m_compiler->filesystemFriendlyName(_contract) + suffix, output);
 | 
						|
		else
 | 
						|
		{
 | 
						|
			cout << title << endl;
 | 
						|
			cout << output << endl;
 | 
						|
		}
 | 
						|
 | 
						|
	}
 | 
						|
}
 | 
						|
 | 
						|
void CommandLineInterface::handleGasEstimation(string const& _contract)
 | 
						|
{
 | 
						|
	Json::Value estimates = m_compiler->gasEstimates(_contract);
 | 
						|
	cout << "Gas estimation:" << endl;
 | 
						|
 | 
						|
	if (estimates["creation"].isObject())
 | 
						|
	{
 | 
						|
		Json::Value creation = estimates["creation"];
 | 
						|
		cout << "construction:" << endl;
 | 
						|
		cout << "   " << creation["executionCost"].asString();
 | 
						|
		cout << " + " << creation["codeDepositCost"].asString();
 | 
						|
		cout << " = " << creation["totalCost"].asString() << endl;
 | 
						|
	}
 | 
						|
 | 
						|
	if (estimates["external"].isObject())
 | 
						|
	{
 | 
						|
		Json::Value externalFunctions = estimates["external"];
 | 
						|
		cout << "external:" << endl;
 | 
						|
		for (auto const& name: externalFunctions.getMemberNames())
 | 
						|
		{
 | 
						|
			if (name.empty())
 | 
						|
				cout << "   fallback:\t";
 | 
						|
			else
 | 
						|
				cout << "   " << name << ":\t";
 | 
						|
			cout << externalFunctions[name].asString() << endl;
 | 
						|
		}
 | 
						|
	}
 | 
						|
 | 
						|
	if (estimates["internal"].isObject())
 | 
						|
	{
 | 
						|
		Json::Value internalFunctions = estimates["internal"];
 | 
						|
		cout << "internal:" << endl;
 | 
						|
		for (auto const& name: internalFunctions.getMemberNames())
 | 
						|
		{
 | 
						|
			cout << "   " << name << ":\t";
 | 
						|
			cout << internalFunctions[name].asString() << endl;
 | 
						|
		}
 | 
						|
	}
 | 
						|
}
 | 
						|
 | 
						|
bool CommandLineInterface::readInputFilesAndConfigureRemappings()
 | 
						|
{
 | 
						|
	bool ignoreMissing = m_args.count(g_argIgnoreMissingFiles);
 | 
						|
	bool addStdin = false;
 | 
						|
	if (!m_args.count(g_argInputFile))
 | 
						|
		addStdin = true;
 | 
						|
	else
 | 
						|
		for (string path: m_args[g_argInputFile].as<vector<string>>())
 | 
						|
		{
 | 
						|
			auto eq = find(path.begin(), path.end(), '=');
 | 
						|
			if (eq != path.end())
 | 
						|
				path = string(eq + 1, path.end());
 | 
						|
			else if (path == "-")
 | 
						|
				addStdin = true;
 | 
						|
			else
 | 
						|
			{
 | 
						|
				auto infile = boost::filesystem::path(path);
 | 
						|
				if (!boost::filesystem::exists(infile))
 | 
						|
				{
 | 
						|
					if (!ignoreMissing)
 | 
						|
					{
 | 
						|
						cerr << "\"" << infile << "\" is not found" << endl;
 | 
						|
						return false;
 | 
						|
					}
 | 
						|
					else
 | 
						|
						cerr << "\"" << infile << "\" is not found. Skipping." << endl;
 | 
						|
 | 
						|
					continue;
 | 
						|
				}
 | 
						|
 | 
						|
				if (!boost::filesystem::is_regular_file(infile))
 | 
						|
				{
 | 
						|
					if (!ignoreMissing)
 | 
						|
					{
 | 
						|
						cerr << "\"" << infile << "\" is not a valid file" << endl;
 | 
						|
						return false;
 | 
						|
					}
 | 
						|
					else
 | 
						|
						cerr << "\"" << infile << "\" is not a valid file. Skipping." << endl;
 | 
						|
 | 
						|
					continue;
 | 
						|
				}
 | 
						|
 | 
						|
				m_sourceCodes[infile.string()] = dev::readFileAsString(infile.string());
 | 
						|
				path = boost::filesystem::canonical(infile).string();
 | 
						|
			}
 | 
						|
			m_allowedDirectories.push_back(boost::filesystem::path(path).remove_filename());
 | 
						|
		}
 | 
						|
	if (addStdin)
 | 
						|
		m_sourceCodes[g_stdinFileName] = dev::readStandardInput();
 | 
						|
 | 
						|
	return true;
 | 
						|
}
 | 
						|
 | 
						|
bool CommandLineInterface::parseLibraryOption(string const& _input)
 | 
						|
{
 | 
						|
	namespace fs = boost::filesystem;
 | 
						|
	string data = _input;
 | 
						|
	try
 | 
						|
	{
 | 
						|
		if (fs::is_regular_file(_input))
 | 
						|
			data = readFileAsString(_input);
 | 
						|
	}
 | 
						|
	catch (fs::filesystem_error const&)
 | 
						|
	{
 | 
						|
		// Thrown e.g. if path is too long.
 | 
						|
	}
 | 
						|
 | 
						|
	vector<string> libraries;
 | 
						|
	boost::split(libraries, data, boost::is_space() || boost::is_any_of(","), boost::token_compress_on);
 | 
						|
	for (string const& lib: libraries)
 | 
						|
		if (!lib.empty())
 | 
						|
		{
 | 
						|
			//search for last colon in string as our binaries output placeholders in the form of file:Name
 | 
						|
			//so we need to search for the second `:` in the string
 | 
						|
			auto colon = lib.rfind(':');
 | 
						|
			if (colon == string::npos)
 | 
						|
			{
 | 
						|
				cerr << "Colon separator missing in library address specifier \"" << lib << "\"" << endl;
 | 
						|
				return false;
 | 
						|
			}
 | 
						|
			string libName(lib.begin(), lib.begin() + colon);
 | 
						|
			string addrString(lib.begin() + colon + 1, lib.end());
 | 
						|
			boost::trim(libName);
 | 
						|
			boost::trim(addrString);
 | 
						|
			if (!passesAddressChecksum(addrString, false))
 | 
						|
			{
 | 
						|
				cerr << "Invalid checksum on library address \"" << libName << "\": " << addrString << endl;
 | 
						|
				return false;
 | 
						|
			}
 | 
						|
			bytes binAddr = fromHex(addrString);
 | 
						|
			h160 address(binAddr, h160::AlignRight);
 | 
						|
			if (binAddr.size() > 20 || address == h160())
 | 
						|
			{
 | 
						|
				cerr << "Invalid address for library \"" << libName << "\": " << addrString << endl;
 | 
						|
				return false;
 | 
						|
			}
 | 
						|
			m_libraries[libName] = address;
 | 
						|
		}
 | 
						|
 | 
						|
	return true;
 | 
						|
}
 | 
						|
 | 
						|
void CommandLineInterface::createFile(string const& _fileName, string const& _data)
 | 
						|
{
 | 
						|
	namespace fs = boost::filesystem;
 | 
						|
	// create directory if not existent
 | 
						|
	fs::path p(m_args.at(g_argOutputDir).as<string>());
 | 
						|
	// Do not try creating the directory if the first item is . or ..
 | 
						|
	if (p.filename() != "." && p.filename() != "..")
 | 
						|
		fs::create_directories(p);
 | 
						|
	string pathName = (p / _fileName).string();
 | 
						|
	if (fs::exists(pathName) && !m_args.count(g_strOverwrite))
 | 
						|
	{
 | 
						|
		cerr << "Refusing to overwrite existing file \"" << pathName << "\" (use --overwrite to force)." << endl;
 | 
						|
		m_error = true;
 | 
						|
		return;
 | 
						|
	}
 | 
						|
	ofstream outFile(pathName);
 | 
						|
	outFile << _data;
 | 
						|
	if (!outFile)
 | 
						|
		BOOST_THROW_EXCEPTION(FileError() << errinfo_comment("Could not write to file: " + pathName));
 | 
						|
}
 | 
						|
 | 
						|
void CommandLineInterface::createJson(string const& _fileName, string const& _json)
 | 
						|
{
 | 
						|
	createFile(boost::filesystem::basename(_fileName) + string(".json"), _json);
 | 
						|
}
 | 
						|
 | 
						|
bool CommandLineInterface::parseArguments(int _argc, char** _argv)
 | 
						|
{
 | 
						|
	// Declare the supported options.
 | 
						|
	po::options_description desc(R"(solc, the Solidity commandline compiler.
 | 
						|
 | 
						|
This program comes with ABSOLUTELY NO WARRANTY. This is free software, and you
 | 
						|
are welcome to redistribute it under certain conditions. See 'solc --license'
 | 
						|
for details.
 | 
						|
 | 
						|
Usage: solc [options] [input_file...]
 | 
						|
Compiles the given Solidity input files (or the standard input if none given or
 | 
						|
"-" is used as a file name) and outputs the components specified in the options
 | 
						|
at standard output or in files in the output directory, if specified.
 | 
						|
Imports are automatically read from the filesystem, but it is also possible to
 | 
						|
remap paths using the context:prefix=path syntax.
 | 
						|
Example:
 | 
						|
    solc --bin -o /tmp/solcoutput dapp-bin=/usr/local/lib/dapp-bin contract.sol
 | 
						|
 | 
						|
Allowed options)",
 | 
						|
		po::options_description::m_default_line_length,
 | 
						|
		po::options_description::m_default_line_length - 23
 | 
						|
	);
 | 
						|
	desc.add_options()
 | 
						|
		(g_argHelp.c_str(), "Show help message and exit.")
 | 
						|
		(g_argVersion.c_str(), "Show version and exit.")
 | 
						|
		(g_strLicense.c_str(), "Show licensing information and exit.")
 | 
						|
		(
 | 
						|
			g_strEVMVersion.c_str(),
 | 
						|
			po::value<string>()->value_name("version"),
 | 
						|
			"Select desired EVM version. Either homestead, tangerineWhistle, spuriousDragon, byzantium (default) or constantinople."
 | 
						|
		)
 | 
						|
		(g_argOptimize.c_str(), "Enable bytecode optimizer.")
 | 
						|
		(
 | 
						|
			g_argOptimizeRuns.c_str(),
 | 
						|
			po::value<unsigned>()->value_name("n")->default_value(200),
 | 
						|
			"Estimated number of contract runs for optimizer tuning."
 | 
						|
		)
 | 
						|
		(g_argPrettyJson.c_str(), "Output JSON in pretty format. Currently it only works with the combined JSON output.")
 | 
						|
		(
 | 
						|
			g_argLibraries.c_str(),
 | 
						|
			po::value<vector<string>>()->value_name("libs"),
 | 
						|
			"Direct string or file containing library addresses. Syntax: "
 | 
						|
			"<libraryName>: <address> [, or whitespace] ...\n"
 | 
						|
			"Address is interpreted as a hex string optionally prefixed by 0x."
 | 
						|
		)
 | 
						|
		(
 | 
						|
			(g_argOutputDir + ",o").c_str(),
 | 
						|
			po::value<string>()->value_name("path"),
 | 
						|
			"If given, creates one file per component and contract/file at the specified directory."
 | 
						|
		)
 | 
						|
		(g_strOverwrite.c_str(), "Overwrite existing files (used together with -o).")
 | 
						|
		(
 | 
						|
			g_argCombinedJson.c_str(),
 | 
						|
			po::value<string>()->value_name(boost::join(g_combinedJsonArgs, ",")),
 | 
						|
			"Output a single json document containing the specified information."
 | 
						|
		)
 | 
						|
		(g_argGas.c_str(), "Print an estimate of the maximal gas usage for each function.")
 | 
						|
		(
 | 
						|
			g_argStandardJSON.c_str(),
 | 
						|
			"Switch to Standard JSON input / output mode, ignoring all options. "
 | 
						|
			"It reads from standard input and provides the result on the standard output."
 | 
						|
		)
 | 
						|
		(
 | 
						|
			g_argAssemble.c_str(),
 | 
						|
			"Switch to assembly mode, ignoring all options except --machine and assumes input is assembly."
 | 
						|
		)
 | 
						|
		(
 | 
						|
			g_argJulia.c_str(),
 | 
						|
			"Switch to JULIA mode, ignoring all options except --machine and assumes input is JULIA."
 | 
						|
		)
 | 
						|
		(
 | 
						|
			g_argStrictAssembly.c_str(),
 | 
						|
			"Switch to strict assembly mode, ignoring all options except --machine and assumes input is strict assembly."
 | 
						|
		)
 | 
						|
		(
 | 
						|
			g_argMachine.c_str(),
 | 
						|
			po::value<string>()->value_name(boost::join(g_machineArgs, ",")),
 | 
						|
			"Target machine in assembly or JULIA mode."
 | 
						|
		)
 | 
						|
		(
 | 
						|
			g_argLink.c_str(),
 | 
						|
			"Switch to linker mode, ignoring all options apart from --libraries "
 | 
						|
			"and modify binaries in place."
 | 
						|
		)
 | 
						|
		(g_argMetadataLiteral.c_str(), "Store referenced sources are literal data in the metadata output.")
 | 
						|
		(
 | 
						|
			g_argAllowPaths.c_str(),
 | 
						|
			po::value<string>()->value_name("path(s)"),
 | 
						|
			"Allow a given path for imports. A list of paths can be supplied by separating them with a comma."
 | 
						|
		)
 | 
						|
		(g_argIgnoreMissingFiles.c_str(), "Ignore missing files.");
 | 
						|
	po::options_description outputComponents("Output Components");
 | 
						|
	outputComponents.add_options()
 | 
						|
		(g_argAst.c_str(), "AST of all source files.")
 | 
						|
		(g_argAstJson.c_str(), "AST of all source files in JSON format.")
 | 
						|
		(g_argAstCompactJson.c_str(), "AST of all source files in a compact JSON format.")
 | 
						|
		(g_argAsm.c_str(), "EVM assembly of the contracts.")
 | 
						|
		(g_argAsmJson.c_str(), "EVM assembly of the contracts in JSON format.")
 | 
						|
		(g_argOpcodes.c_str(), "Opcodes of the contracts.")
 | 
						|
		(g_argBinary.c_str(), "Binary of the contracts in hex.")
 | 
						|
		(g_argBinaryRuntime.c_str(), "Binary of the runtime part of the contracts in hex.")
 | 
						|
		(g_argCloneBinary.c_str(), "Binary of the clone contracts in hex.")
 | 
						|
		(g_argAbi.c_str(), "ABI specification of the contracts.")
 | 
						|
		(g_argSignatureHashes.c_str(), "Function signature hashes of the contracts.")
 | 
						|
		(g_argNatspecUser.c_str(), "Natspec user documentation of all contracts.")
 | 
						|
		(g_argNatspecDev.c_str(), "Natspec developer documentation of all contracts.")
 | 
						|
		(g_argMetadata.c_str(), "Combined Metadata JSON whose Swarm hash is stored on-chain.")
 | 
						|
		(g_argFormal.c_str(), "Translated source suitable for formal analysis. (Deprecated)");
 | 
						|
	desc.add(outputComponents);
 | 
						|
 | 
						|
	po::options_description allOptions = desc;
 | 
						|
	allOptions.add_options()(g_argInputFile.c_str(), po::value<vector<string>>(), "input file");
 | 
						|
 | 
						|
	// All positional options should be interpreted as input files
 | 
						|
	po::positional_options_description filesPositions;
 | 
						|
	filesPositions.add(g_argInputFile.c_str(), -1);
 | 
						|
 | 
						|
	// parse the compiler arguments
 | 
						|
	try
 | 
						|
	{
 | 
						|
		po::command_line_parser cmdLineParser(_argc, _argv);
 | 
						|
		cmdLineParser.style(po::command_line_style::default_style & (~po::command_line_style::allow_guessing));
 | 
						|
		cmdLineParser.options(allOptions).positional(filesPositions);
 | 
						|
		po::store(cmdLineParser.run(), m_args);
 | 
						|
	}
 | 
						|
	catch (po::error const& _exception)
 | 
						|
	{
 | 
						|
		cerr << _exception.what() << endl;
 | 
						|
		return false;
 | 
						|
	}
 | 
						|
 | 
						|
	if (m_args.count(g_argHelp) || (isatty(fileno(stdin)) && _argc == 1))
 | 
						|
	{
 | 
						|
		cout << desc;
 | 
						|
		return false;
 | 
						|
	}
 | 
						|
 | 
						|
	if (m_args.count(g_argVersion))
 | 
						|
	{
 | 
						|
		version();
 | 
						|
		return false;
 | 
						|
	}
 | 
						|
 | 
						|
	if (m_args.count(g_strLicense))
 | 
						|
	{
 | 
						|
		license();
 | 
						|
		return false;
 | 
						|
	}
 | 
						|
 | 
						|
	if (m_args.count(g_argCombinedJson))
 | 
						|
	{
 | 
						|
		vector<string> requests;
 | 
						|
		for (string const& item: boost::split(requests, m_args[g_argCombinedJson].as<string>(), boost::is_any_of(",")))
 | 
						|
			if (!g_combinedJsonArgs.count(item))
 | 
						|
			{
 | 
						|
				cerr << "Invalid option to --combined-json: " << item << endl;
 | 
						|
				return false;
 | 
						|
			}
 | 
						|
	}
 | 
						|
	po::notify(m_args);
 | 
						|
 | 
						|
	return true;
 | 
						|
}
 | 
						|
 | 
						|
bool CommandLineInterface::processInput()
 | 
						|
{
 | 
						|
	ReadCallback::Callback fileReader = [this](string const& _path)
 | 
						|
	{
 | 
						|
		try
 | 
						|
		{
 | 
						|
			auto path = boost::filesystem::path(_path);
 | 
						|
			auto canonicalPath = weaklyCanonicalFilesystemPath(path);
 | 
						|
			bool isAllowed = false;
 | 
						|
			for (auto const& allowedDir: m_allowedDirectories)
 | 
						|
			{
 | 
						|
				// If dir is a prefix of boostPath, we are fine.
 | 
						|
				if (
 | 
						|
					std::distance(allowedDir.begin(), allowedDir.end()) <= std::distance(canonicalPath.begin(), canonicalPath.end()) &&
 | 
						|
					std::equal(allowedDir.begin(), allowedDir.end(), canonicalPath.begin())
 | 
						|
				)
 | 
						|
				{
 | 
						|
					isAllowed = true;
 | 
						|
					break;
 | 
						|
				}
 | 
						|
			}
 | 
						|
			if (!isAllowed)
 | 
						|
				return ReadCallback::Result{false, "File outside of allowed directories."};
 | 
						|
 | 
						|
			if (!boost::filesystem::exists(canonicalPath))
 | 
						|
				return ReadCallback::Result{false, "File not found."};
 | 
						|
 | 
						|
			if (!boost::filesystem::is_regular_file(canonicalPath))
 | 
						|
				return ReadCallback::Result{false, "Not a valid file."};
 | 
						|
 | 
						|
			auto contents = dev::readFileAsString(canonicalPath.string());
 | 
						|
			m_sourceCodes[path.string()] = contents;
 | 
						|
			return ReadCallback::Result{true, contents};
 | 
						|
		}
 | 
						|
		catch (Exception const& _exception)
 | 
						|
		{
 | 
						|
			return ReadCallback::Result{false, "Exception in read callback: " + boost::diagnostic_information(_exception)};
 | 
						|
		}
 | 
						|
		catch (...)
 | 
						|
		{
 | 
						|
			return ReadCallback::Result{false, "Unknown exception in read callback."};
 | 
						|
		}
 | 
						|
	};
 | 
						|
 | 
						|
	if (m_args.count(g_argAllowPaths))
 | 
						|
	{
 | 
						|
		vector<string> paths;
 | 
						|
		for (string const& path: boost::split(paths, m_args[g_argAllowPaths].as<string>(), boost::is_any_of(","))) {
 | 
						|
			auto filesystem_path = boost::filesystem::path(path);
 | 
						|
			// If the given path had a trailing slash, the Boost filesystem
 | 
						|
			// path will have it's last component set to '.'. This breaks
 | 
						|
			// path comparison in later parts of the code, so we need to strip
 | 
						|
			// it.
 | 
						|
			if (filesystem_path.filename() == ".") {
 | 
						|
				filesystem_path.remove_filename();
 | 
						|
			}
 | 
						|
			m_allowedDirectories.push_back(filesystem_path);
 | 
						|
		}
 | 
						|
	}
 | 
						|
 | 
						|
	if (m_args.count(g_argStandardJSON))
 | 
						|
	{
 | 
						|
		string input = dev::readStandardInput();
 | 
						|
		StandardCompiler compiler(fileReader);
 | 
						|
		cout << compiler.compile(input) << endl;
 | 
						|
		return true;
 | 
						|
	}
 | 
						|
 | 
						|
	if (!readInputFilesAndConfigureRemappings())
 | 
						|
		return false;
 | 
						|
 | 
						|
	if (m_args.count(g_argLibraries))
 | 
						|
		for (string const& library: m_args[g_argLibraries].as<vector<string>>())
 | 
						|
			if (!parseLibraryOption(library))
 | 
						|
				return false;
 | 
						|
 | 
						|
	if (m_args.count(g_strEVMVersion))
 | 
						|
	{
 | 
						|
		string versionOptionStr = m_args[g_strEVMVersion].as<string>();
 | 
						|
		boost::optional<EVMVersion> versionOption = EVMVersion::fromString(versionOptionStr);
 | 
						|
		if (!versionOption)
 | 
						|
		{
 | 
						|
			cerr << "Invalid option for --evm-version: " << versionOptionStr << endl;
 | 
						|
			return false;
 | 
						|
		}
 | 
						|
		m_evmVersion = *versionOption;
 | 
						|
	}
 | 
						|
 | 
						|
	if (m_args.count(g_argAssemble) || m_args.count(g_argStrictAssembly) || m_args.count(g_argJulia))
 | 
						|
	{
 | 
						|
		// switch to assembly mode
 | 
						|
		m_onlyAssemble = true;
 | 
						|
		using Input = AssemblyStack::Language;
 | 
						|
		using Machine = AssemblyStack::Machine;
 | 
						|
		Input inputLanguage = m_args.count(g_argJulia) ? Input::JULIA : (m_args.count(g_argStrictAssembly) ? Input::StrictAssembly : Input::Assembly);
 | 
						|
		Machine targetMachine = Machine::EVM;
 | 
						|
		if (m_args.count(g_argMachine))
 | 
						|
		{
 | 
						|
			string machine = m_args[g_argMachine].as<string>();
 | 
						|
			if (machine == g_strEVM)
 | 
						|
				targetMachine = Machine::EVM;
 | 
						|
			else if (machine == g_strEVM15)
 | 
						|
				targetMachine = Machine::EVM15;
 | 
						|
			else if (machine == g_streWasm)
 | 
						|
				targetMachine = Machine::eWasm;
 | 
						|
			else
 | 
						|
			{
 | 
						|
				cerr << "Invalid option for --machine: " << machine << endl;
 | 
						|
				return false;
 | 
						|
			}
 | 
						|
		}
 | 
						|
		return assemble(inputLanguage, targetMachine);
 | 
						|
	}
 | 
						|
	if (m_args.count(g_argLink))
 | 
						|
	{
 | 
						|
		// switch to linker mode
 | 
						|
		m_onlyLink = true;
 | 
						|
		return link();
 | 
						|
	}
 | 
						|
 | 
						|
	m_compiler.reset(new CompilerStack(fileReader));
 | 
						|
 | 
						|
	auto scannerFromSourceName = [&](string const& _sourceName) -> solidity::Scanner const& { return m_compiler->scanner(_sourceName); };
 | 
						|
	SourceReferenceFormatter formatter(cerr, scannerFromSourceName);
 | 
						|
 | 
						|
	try
 | 
						|
	{
 | 
						|
		if (m_args.count(g_argMetadataLiteral) > 0)
 | 
						|
			m_compiler->useMetadataLiteralSources(true);
 | 
						|
		if (m_args.count(g_argInputFile))
 | 
						|
			m_compiler->setRemappings(m_args[g_argInputFile].as<vector<string>>());
 | 
						|
		for (auto const& sourceCode: m_sourceCodes)
 | 
						|
			m_compiler->addSource(sourceCode.first, sourceCode.second);
 | 
						|
		if (m_args.count(g_argLibraries))
 | 
						|
			m_compiler->setLibraries(m_libraries);
 | 
						|
		m_compiler->setEVMVersion(m_evmVersion);
 | 
						|
		// TODO: Perhaps we should not compile unless requested
 | 
						|
		bool optimize = m_args.count(g_argOptimize) > 0;
 | 
						|
		unsigned runs = m_args[g_argOptimizeRuns].as<unsigned>();
 | 
						|
		m_compiler->setOptimiserSettings(optimize, runs);
 | 
						|
 | 
						|
		bool successful = m_compiler->compile();
 | 
						|
 | 
						|
		for (auto const& error: m_compiler->errors())
 | 
						|
			formatter.printExceptionInformation(
 | 
						|
				*error,
 | 
						|
				(error->type() == Error::Type::Warning) ? "Warning" : "Error"
 | 
						|
			);
 | 
						|
 | 
						|
		if (!successful)
 | 
						|
			return false;
 | 
						|
	}
 | 
						|
	catch (CompilerError const& _exception)
 | 
						|
	{
 | 
						|
		formatter.printExceptionInformation(_exception, "Compiler error");
 | 
						|
		return false;
 | 
						|
	}
 | 
						|
	catch (InternalCompilerError const& _exception)
 | 
						|
	{
 | 
						|
		cerr << "Internal compiler error during compilation:" << endl
 | 
						|
			 << boost::diagnostic_information(_exception);
 | 
						|
		return false;
 | 
						|
	}
 | 
						|
	catch (UnimplementedFeatureError const& _exception)
 | 
						|
	{
 | 
						|
		cerr << "Unimplemented feature:" << endl
 | 
						|
			 << boost::diagnostic_information(_exception);
 | 
						|
		return false;
 | 
						|
	}
 | 
						|
	catch (Error const& _error)
 | 
						|
	{
 | 
						|
		if (_error.type() == Error::Type::DocstringParsingError)
 | 
						|
			cerr << "Documentation parsing error: " << *boost::get_error_info<errinfo_comment>(_error) << endl;
 | 
						|
		else
 | 
						|
			formatter.printExceptionInformation(_error, _error.typeName());
 | 
						|
 | 
						|
		return false;
 | 
						|
	}
 | 
						|
	catch (Exception const& _exception)
 | 
						|
	{
 | 
						|
		cerr << "Exception during compilation: " << boost::diagnostic_information(_exception) << endl;
 | 
						|
		return false;
 | 
						|
	}
 | 
						|
	catch (...)
 | 
						|
	{
 | 
						|
		cerr << "Unknown exception during compilation." << endl;
 | 
						|
		return false;
 | 
						|
	}
 | 
						|
 | 
						|
	return true;
 | 
						|
}
 | 
						|
 | 
						|
void CommandLineInterface::handleCombinedJSON()
 | 
						|
{
 | 
						|
	if (!m_args.count(g_argCombinedJson))
 | 
						|
		return;
 | 
						|
 | 
						|
	Json::Value output(Json::objectValue);
 | 
						|
 | 
						|
	output[g_strVersion] = ::dev::solidity::VersionString;
 | 
						|
	set<string> requests;
 | 
						|
	boost::split(requests, m_args[g_argCombinedJson].as<string>(), boost::is_any_of(","));
 | 
						|
	vector<string> contracts = m_compiler->contractNames();
 | 
						|
 | 
						|
	if (!contracts.empty())
 | 
						|
		output[g_strContracts] = Json::Value(Json::objectValue);
 | 
						|
	for (string const& contractName: contracts)
 | 
						|
	{
 | 
						|
		Json::Value& contractData = output[g_strContracts][contractName] = Json::objectValue;
 | 
						|
		if (requests.count(g_strAbi))
 | 
						|
			contractData[g_strAbi] = dev::jsonCompactPrint(m_compiler->contractABI(contractName));
 | 
						|
		if (requests.count("metadata"))
 | 
						|
			contractData["metadata"] = m_compiler->metadata(contractName);
 | 
						|
		if (requests.count(g_strBinary))
 | 
						|
			contractData[g_strBinary] = m_compiler->object(contractName).toHex();
 | 
						|
		if (requests.count(g_strBinaryRuntime))
 | 
						|
			contractData[g_strBinaryRuntime] = m_compiler->runtimeObject(contractName).toHex();
 | 
						|
		if (requests.count(g_strCloneBinary))
 | 
						|
			contractData[g_strCloneBinary] = m_compiler->cloneObject(contractName).toHex();
 | 
						|
		if (requests.count(g_strOpcodes))
 | 
						|
			contractData[g_strOpcodes] = solidity::disassemble(m_compiler->object(contractName).bytecode);
 | 
						|
		if (requests.count(g_strAsm))
 | 
						|
			contractData[g_strAsm] = m_compiler->assemblyJSON(contractName, m_sourceCodes);
 | 
						|
		if (requests.count(g_strSrcMap))
 | 
						|
		{
 | 
						|
			auto map = m_compiler->sourceMapping(contractName);
 | 
						|
			contractData[g_strSrcMap] = map ? *map : "";
 | 
						|
		}
 | 
						|
		if (requests.count(g_strSrcMapRuntime))
 | 
						|
		{
 | 
						|
			auto map = m_compiler->runtimeSourceMapping(contractName);
 | 
						|
			contractData[g_strSrcMapRuntime] = map ? *map : "";
 | 
						|
		}
 | 
						|
		if (requests.count(g_strSignatureHashes))
 | 
						|
			contractData[g_strSignatureHashes] = m_compiler->methodIdentifiers(contractName);
 | 
						|
		if (requests.count(g_strNatspecDev))
 | 
						|
			contractData[g_strNatspecDev] = dev::jsonCompactPrint(m_compiler->natspecDev(contractName));
 | 
						|
		if (requests.count(g_strNatspecUser))
 | 
						|
			contractData[g_strNatspecUser] = dev::jsonCompactPrint(m_compiler->natspecUser(contractName));
 | 
						|
	}
 | 
						|
 | 
						|
	bool needsSourceList = requests.count(g_strAst) || requests.count(g_strSrcMap) || requests.count(g_strSrcMapRuntime);
 | 
						|
	if (needsSourceList)
 | 
						|
	{
 | 
						|
		// Indices into this array are used to abbreviate source names in source locations.
 | 
						|
		output[g_strSourceList] = Json::Value(Json::arrayValue);
 | 
						|
 | 
						|
		for (auto const& source: m_compiler->sourceNames())
 | 
						|
			output[g_strSourceList].append(source);
 | 
						|
	}
 | 
						|
 | 
						|
	if (requests.count(g_strAst))
 | 
						|
	{
 | 
						|
		bool legacyFormat = !requests.count(g_strCompactJSON);
 | 
						|
		output[g_strSources] = Json::Value(Json::objectValue);
 | 
						|
		for (auto const& sourceCode: m_sourceCodes)
 | 
						|
		{
 | 
						|
			ASTJsonConverter converter(legacyFormat, m_compiler->sourceIndices());
 | 
						|
			output[g_strSources][sourceCode.first] = Json::Value(Json::objectValue);
 | 
						|
			output[g_strSources][sourceCode.first]["AST"] = converter.toJson(m_compiler->ast(sourceCode.first));
 | 
						|
		}
 | 
						|
	}
 | 
						|
 | 
						|
	string json = m_args.count(g_argPrettyJson) ? dev::jsonPrettyPrint(output) : dev::jsonCompactPrint(output);
 | 
						|
 | 
						|
	if (m_args.count(g_argOutputDir))
 | 
						|
		createJson("combined", json);
 | 
						|
	else
 | 
						|
		cout << json << endl;
 | 
						|
}
 | 
						|
 | 
						|
void CommandLineInterface::handleAst(string const& _argStr)
 | 
						|
{
 | 
						|
	string title;
 | 
						|
 | 
						|
	if (_argStr == g_argAst)
 | 
						|
		title = "Syntax trees:";
 | 
						|
	else if (_argStr == g_argAstJson)
 | 
						|
		title = "JSON AST:";
 | 
						|
	else if (_argStr == g_argAstCompactJson)
 | 
						|
		title = "JSON AST (compact format):";
 | 
						|
	else
 | 
						|
		BOOST_THROW_EXCEPTION(InternalCompilerError() << errinfo_comment("Illegal argStr for AST"));
 | 
						|
 | 
						|
	// do we need AST output?
 | 
						|
	if (m_args.count(_argStr))
 | 
						|
	{
 | 
						|
		vector<ASTNode const*> asts;
 | 
						|
		for (auto const& sourceCode: m_sourceCodes)
 | 
						|
			asts.push_back(&m_compiler->ast(sourceCode.first));
 | 
						|
		map<ASTNode const*, eth::GasMeter::GasConsumption> gasCosts;
 | 
						|
		// FIXME: shouldn't this be done for every contract?
 | 
						|
		if (m_compiler->runtimeAssemblyItems(m_compiler->lastContractName()))
 | 
						|
			gasCosts = GasEstimator::breakToStatementLevel(
 | 
						|
				GasEstimator(m_evmVersion).structuralEstimation(*m_compiler->runtimeAssemblyItems(m_compiler->lastContractName()), asts),
 | 
						|
				asts
 | 
						|
			);
 | 
						|
 | 
						|
		bool legacyFormat = !m_args.count(g_argAstCompactJson);
 | 
						|
		if (m_args.count(g_argOutputDir))
 | 
						|
		{
 | 
						|
			for (auto const& sourceCode: m_sourceCodes)
 | 
						|
			{
 | 
						|
				stringstream data;
 | 
						|
				string postfix = "";
 | 
						|
				if (_argStr == g_argAst)
 | 
						|
				{
 | 
						|
					ASTPrinter printer(m_compiler->ast(sourceCode.first), sourceCode.second);
 | 
						|
					printer.print(data);
 | 
						|
				}
 | 
						|
				else
 | 
						|
				{
 | 
						|
					ASTJsonConverter(legacyFormat, m_compiler->sourceIndices()).print(data, m_compiler->ast(sourceCode.first));
 | 
						|
					postfix += "_json";
 | 
						|
				}
 | 
						|
				boost::filesystem::path path(sourceCode.first);
 | 
						|
				createFile(path.filename().string() + postfix + ".ast", data.str());
 | 
						|
			}
 | 
						|
		}
 | 
						|
		else
 | 
						|
		{
 | 
						|
			cout << title << endl << endl;
 | 
						|
			for (auto const& sourceCode: m_sourceCodes)
 | 
						|
			{
 | 
						|
				cout << endl << "======= " << sourceCode.first << " =======" << endl;
 | 
						|
				if (_argStr == g_argAst)
 | 
						|
				{
 | 
						|
					ASTPrinter printer(
 | 
						|
						m_compiler->ast(sourceCode.first),
 | 
						|
						sourceCode.second,
 | 
						|
						gasCosts
 | 
						|
					);
 | 
						|
					printer.print(cout);
 | 
						|
				}
 | 
						|
				else
 | 
						|
					ASTJsonConverter(legacyFormat, m_compiler->sourceIndices()).print(cout, m_compiler->ast(sourceCode.first));
 | 
						|
			}
 | 
						|
		}
 | 
						|
	}
 | 
						|
}
 | 
						|
 | 
						|
bool CommandLineInterface::actOnInput()
 | 
						|
{
 | 
						|
	if (m_args.count(g_argStandardJSON) || m_onlyAssemble)
 | 
						|
		// Already done in "processInput" phase.
 | 
						|
		return true;
 | 
						|
	else if (m_onlyLink)
 | 
						|
		writeLinkedFiles();
 | 
						|
	else
 | 
						|
		outputCompilationResults();
 | 
						|
	return !m_error;
 | 
						|
}
 | 
						|
 | 
						|
bool CommandLineInterface::link()
 | 
						|
{
 | 
						|
	// Map from how the libraries will be named inside the bytecode to their addresses.
 | 
						|
	map<string, h160> librariesReplacements;
 | 
						|
	int const placeholderSize = 40; // 20 bytes or 40 hex characters
 | 
						|
	for (auto const& library: m_libraries)
 | 
						|
	{
 | 
						|
		string const& name = library.first;
 | 
						|
		// Library placeholders are 40 hex digits (20 bytes) that start and end with '__'.
 | 
						|
		// This leaves 36 characters for the library name, while too short library names are
 | 
						|
		// padded on the right with '_' and too long names are truncated.
 | 
						|
		string replacement = "__";
 | 
						|
		for (size_t i = 0; i < placeholderSize - 4; ++i)
 | 
						|
			replacement.push_back(i < name.size() ? name[i] : '_');
 | 
						|
		replacement += "__";
 | 
						|
		librariesReplacements[replacement] = library.second;
 | 
						|
	}
 | 
						|
	for (auto& src: m_sourceCodes)
 | 
						|
	{
 | 
						|
		auto end = src.second.end();
 | 
						|
		for (auto it = src.second.begin(); it != end;)
 | 
						|
		{
 | 
						|
			while (it != end && *it != '_') ++it;
 | 
						|
			if (it == end) break;
 | 
						|
			if (end - it < placeholderSize)
 | 
						|
			{
 | 
						|
				cerr << "Error in binary object file " << src.first << " at position " << (end - src.second.begin()) << endl;
 | 
						|
				return false;
 | 
						|
			}
 | 
						|
 | 
						|
			string name(it, it + placeholderSize);
 | 
						|
			if (librariesReplacements.count(name))
 | 
						|
			{
 | 
						|
				string hexStr(toHex(librariesReplacements.at(name).asBytes()));
 | 
						|
				copy(hexStr.begin(), hexStr.end(), it);
 | 
						|
			}
 | 
						|
			else
 | 
						|
				cerr << "Reference \"" << name << "\" in file \"" << src.first << "\" still unresolved." << endl;
 | 
						|
			it += placeholderSize;
 | 
						|
		}
 | 
						|
	}
 | 
						|
	return true;
 | 
						|
}
 | 
						|
 | 
						|
void CommandLineInterface::writeLinkedFiles()
 | 
						|
{
 | 
						|
	for (auto const& src: m_sourceCodes)
 | 
						|
		if (src.first == g_stdinFileName)
 | 
						|
			cout << src.second << endl;
 | 
						|
		else
 | 
						|
			writeFile(src.first, src.second);
 | 
						|
}
 | 
						|
 | 
						|
bool CommandLineInterface::assemble(
 | 
						|
	AssemblyStack::Language _language,
 | 
						|
	AssemblyStack::Machine _targetMachine
 | 
						|
)
 | 
						|
{
 | 
						|
	bool successful = true;
 | 
						|
	map<string, AssemblyStack> assemblyStacks;
 | 
						|
	for (auto const& src: m_sourceCodes)
 | 
						|
	{
 | 
						|
		auto& stack = assemblyStacks[src.first] = AssemblyStack(m_evmVersion, _language);
 | 
						|
		try
 | 
						|
		{
 | 
						|
			if (!stack.parseAndAnalyze(src.first, src.second))
 | 
						|
				successful = false;
 | 
						|
		}
 | 
						|
		catch (Exception const& _exception)
 | 
						|
		{
 | 
						|
			cerr << "Exception in assembler: " << boost::diagnostic_information(_exception) << endl;
 | 
						|
			return false;
 | 
						|
		}
 | 
						|
		catch (...)
 | 
						|
		{
 | 
						|
			cerr << "Unknown exception in assembler." << endl;
 | 
						|
			return false;
 | 
						|
		}
 | 
						|
	}
 | 
						|
 | 
						|
	for (auto const& sourceAndStack: assemblyStacks)
 | 
						|
	{
 | 
						|
		auto const& stack = sourceAndStack.second;
 | 
						|
		auto scannerFromSourceName = [&](string const&) -> Scanner const& { return stack.scanner(); };
 | 
						|
		SourceReferenceFormatter formatter(cerr, scannerFromSourceName);
 | 
						|
 | 
						|
		for (auto const& error: stack.errors())
 | 
						|
			formatter.printExceptionInformation(
 | 
						|
				*error,
 | 
						|
				(error->type() == Error::Type::Warning) ? "Warning" : "Error"
 | 
						|
			);
 | 
						|
		if (!Error::containsOnlyWarnings(stack.errors()))
 | 
						|
			successful = false;
 | 
						|
	}
 | 
						|
 | 
						|
	if (!successful)
 | 
						|
		return false;
 | 
						|
 | 
						|
	for (auto const& src: m_sourceCodes)
 | 
						|
	{
 | 
						|
		string machine =
 | 
						|
			_targetMachine == AssemblyStack::Machine::EVM ? "EVM" :
 | 
						|
			_targetMachine == AssemblyStack::Machine::EVM15 ? "EVM 1.5" :
 | 
						|
			"eWasm";
 | 
						|
		cout << endl << "======= " << src.first << " (" << machine << ") =======" << endl;
 | 
						|
		AssemblyStack& stack = assemblyStacks[src.first];
 | 
						|
 | 
						|
		cout << endl << "Pretty printed source:" << endl;
 | 
						|
		cout << stack.print() << endl;
 | 
						|
 | 
						|
		MachineAssemblyObject object;
 | 
						|
		try
 | 
						|
		{
 | 
						|
			object = stack.assemble(_targetMachine);
 | 
						|
		}
 | 
						|
		catch (Exception const& _exception)
 | 
						|
		{
 | 
						|
			cerr << "Exception while assembling: " << boost::diagnostic_information(_exception) << endl;
 | 
						|
			return false;
 | 
						|
		}
 | 
						|
		catch (...)
 | 
						|
		{
 | 
						|
			cerr << "Unknown exception while assembling." << endl;
 | 
						|
			return false;
 | 
						|
		}
 | 
						|
 | 
						|
		cout << endl << "Binary representation:" << endl;
 | 
						|
		if (object.bytecode)
 | 
						|
			cout << object.bytecode->toHex() << endl;
 | 
						|
		else
 | 
						|
			cerr << "No binary representation found." << endl;
 | 
						|
 | 
						|
		cout << endl << "Text representation:" << endl;
 | 
						|
		if (!object.assembly.empty())
 | 
						|
			cout << object.assembly << endl;
 | 
						|
		else
 | 
						|
			cerr << "No text representation found." << endl;
 | 
						|
	}
 | 
						|
 | 
						|
	return true;
 | 
						|
}
 | 
						|
 | 
						|
void CommandLineInterface::outputCompilationResults()
 | 
						|
{
 | 
						|
	handleCombinedJSON();
 | 
						|
 | 
						|
	// do we need AST output?
 | 
						|
	handleAst(g_argAst);
 | 
						|
	handleAst(g_argAstJson);
 | 
						|
	handleAst(g_argAstCompactJson);
 | 
						|
 | 
						|
	vector<string> contracts = m_compiler->contractNames();
 | 
						|
	for (string const& contract: contracts)
 | 
						|
	{
 | 
						|
		if (needsHumanTargetedStdout(m_args))
 | 
						|
			cout << endl << "======= " << contract << " =======" << endl;
 | 
						|
 | 
						|
		// do we need EVM assembly?
 | 
						|
		if (m_args.count(g_argAsm) || m_args.count(g_argAsmJson))
 | 
						|
		{
 | 
						|
			string ret;
 | 
						|
			if (m_args.count(g_argAsmJson))
 | 
						|
				ret = dev::jsonPrettyPrint(m_compiler->assemblyJSON(contract, m_sourceCodes));
 | 
						|
			else
 | 
						|
				ret = m_compiler->assemblyString(contract, m_sourceCodes);
 | 
						|
 | 
						|
			if (m_args.count(g_argOutputDir))
 | 
						|
			{
 | 
						|
				createFile(m_compiler->filesystemFriendlyName(contract) + (m_args.count(g_argAsmJson) ? "_evm.json" : ".evm"), ret);
 | 
						|
			}
 | 
						|
			else
 | 
						|
			{
 | 
						|
				cout << "EVM assembly:" << endl << ret << endl;
 | 
						|
			}
 | 
						|
		}
 | 
						|
 | 
						|
		if (m_args.count(g_argGas))
 | 
						|
			handleGasEstimation(contract);
 | 
						|
 | 
						|
		handleBytecode(contract);
 | 
						|
		handleSignatureHashes(contract);
 | 
						|
		handleMetadata(contract);
 | 
						|
		handleABI(contract);
 | 
						|
		handleNatspec(true, contract);
 | 
						|
		handleNatspec(false, contract);
 | 
						|
	} // end of contracts iteration
 | 
						|
 | 
						|
	if (m_args.count(g_argFormal))
 | 
						|
		cerr << "Support for the Why3 output was removed." << endl;
 | 
						|
}
 | 
						|
 | 
						|
}
 | 
						|
}
 |