solidity/solc/CommandLineInterface.cpp

1326 lines
43 KiB
C++
Raw Normal View History

/*
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
/**
* @author Lefteris <lefteris@ethdev.com>
* @author Gav Wood <g@ethdev.com>
* @date 2014
* Solidity command line interface.
*/
2019-02-13 11:07:20 +00:00
#include <solc/CommandLineInterface.h>
#include <solc/Exceptions.h>
#include "license.h"
2015-09-16 13:55:25 +00:00
#include "solidity/BuildInfo.h"
2017-01-26 12:47:57 +00:00
2015-10-20 22:21:52 +00:00
#include <libsolidity/interface/Version.h>
#include <libsolidity/ast/ASTJsonExporter.h>
2019-09-11 19:16:35 +00:00
#include <libsolidity/ast/ASTJsonImporter.h>
2015-10-20 22:21:52 +00:00
#include <libsolidity/analysis/NameAndTypeResolver.h>
#include <libsolidity/interface/CompilerStack.h>
2017-02-09 14:55:57 +00:00
#include <libsolidity/interface/StandardCompiler.h>
2015-10-20 22:21:52 +00:00
#include <libsolidity/interface/GasEstimator.h>
#include <libsolidity/interface/DebugSettings.h>
#include <libsolidity/interface/ImportRemapper.h>
#include <libsolidity/interface/StorageLayout.h>
#include <libsolidity/lsp/LanguageServer.h>
#include <libsolidity/lsp/Transport.h>
2019-02-13 11:07:20 +00:00
#include <libyul/YulStack.h>
2017-01-26 12:47:57 +00:00
#include <libevmasm/Instruction.h>
#include <libevmasm/Disassemble.h>
2017-01-26 12:47:57 +00:00
#include <libevmasm/GasMeter.h>
2019-02-13 11:07:20 +00:00
#include <liblangutil/Exceptions.h>
2020-12-01 13:22:15 +00:00
#include <liblangutil/SourceReferenceFormatter.h>
2019-02-13 11:07:20 +00:00
2020-05-20 10:55:12 +00:00
#include <libsmtutil/Exceptions.h>
#include <libsolutil/Common.h>
#include <libsolutil/CommonData.h>
#include <libsolutil/CommonIO.h>
#include <libsolutil/JSON.h>
2017-01-26 12:47:57 +00:00
#include <algorithm>
#include <fstream>
#include <memory>
#include <range/v3/view/map.hpp>
2017-01-26 12:47:57 +00:00
#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
2017-01-26 12:47:57 +00:00
#include <fstream>
#if !defined(STDERR_FILENO)
#define STDERR_FILENO 2
#endif
using namespace std::string_literals;
2019-12-11 16:31:36 +00:00
using namespace solidity;
using namespace solidity::util;
using namespace solidity::langutil;
2022-09-28 03:07:35 +00:00
namespace
{
std::set<frontend::InputMode> const CompilerInputModes{
2022-09-28 03:07:35 +00:00
frontend::InputMode::Compiler,
frontend::InputMode::CompilerWithASTImport,
frontend::InputMode::EVMAssemblerJSON
2022-09-28 03:07:35 +00:00
};
} // anonymous namespace
2019-12-11 16:31:36 +00:00
namespace solidity::frontend
{
std::ostream& CommandLineInterface::sout(bool _markAsUsed)
2020-12-08 20:06:10 +00:00
{
if (_markAsUsed)
m_hasOutput = true;
return m_sout;
}
std::ostream& CommandLineInterface::serr(bool _markAsUsed)
{
if (_markAsUsed)
m_hasOutput = true;
return m_serr;
2020-12-08 20:06:10 +00:00
}
#define cin
#define cout
#define cerr
static std::string const g_stdinFileName = "<stdin>";
static std::string const g_strAbi = "abi";
static std::string const g_strAsm = "asm";
static std::string const g_strAst = "ast";
static std::string const g_strBinary = "bin";
static std::string const g_strBinaryRuntime = "bin-runtime";
static std::string const g_strContracts = "contracts";
static std::string const g_strFunDebug = "function-debug";
static std::string const g_strFunDebugRuntime = "function-debug-runtime";
static std::string const g_strGeneratedSources = "generated-sources";
static std::string const g_strGeneratedSourcesRuntime = "generated-sources-runtime";
static std::string const g_strNatspecDev = "devdoc";
static std::string const g_strNatspecUser = "userdoc";
static std::string const g_strOpcodes = "opcodes";
static std::string const g_strSignatureHashes = "hashes";
static std::string const g_strSourceList = "sourceList";
static std::string const g_strSources = "sources";
static std::string const g_strSrcMap = "srcmap";
static std::string const g_strSrcMapRuntime = "srcmap-runtime";
static std::string const g_strStorageLayout = "storage-layout";
static std::string const g_strVersion = "version";
static bool needsHumanTargetedStdout(CommandLineOptions const& _options)
{
if (_options.compiler.estimateGas)
2015-10-21 14:43:31 +00:00
return true;
if (!_options.output.dir.empty())
2015-10-21 14:43:31 +00:00
return false;
return
_options.compiler.outputs.abi ||
_options.compiler.outputs.asm_ ||
_options.compiler.outputs.asmJson ||
_options.compiler.outputs.binary ||
_options.compiler.outputs.binaryRuntime ||
_options.compiler.outputs.metadata ||
_options.compiler.outputs.natspecUser ||
_options.compiler.outputs.natspecDev ||
_options.compiler.outputs.opcodes ||
_options.compiler.outputs.signatureHashes ||
_options.compiler.outputs.storageLayout;
}
static bool coloredOutput(CommandLineOptions const& _options)
2020-07-08 20:08:50 +00:00
{
return
(!_options.formatting.coloredOutput.has_value() && isatty(STDERR_FILENO)) ||
(_options.formatting.coloredOutput.has_value() && _options.formatting.coloredOutput.value());
2020-12-08 20:06:10 +00:00
}
void CommandLineInterface::handleEVMAssembly(std::string const& _contract)
{
solAssert(m_assemblyStack);
solAssert(CompilerInputModes.count(m_options.input.mode) == 1);
if (m_options.compiler.outputs.asm_ || m_options.compiler.outputs.asmJson)
{
std::string assembly;
if (m_options.compiler.outputs.asmJson)
assembly = util::jsonPrint(removeNullMembers(m_assemblyStack->assemblyJSON(_contract)), m_options.formatting.json);
else
assembly = m_assemblyStack->assemblyString(_contract, m_fileReader.sourceUnits());
if (!m_options.output.dir.empty())
createFile(
m_compiler->filesystemFriendlyName(_contract) +
(m_options.compiler.outputs.asmJson ? "_evm.json" : ".evm"),
assembly
);
else
sout() << "EVM assembly:" << std::endl << assembly << std::endl;
}
}
void CommandLineInterface::handleBinary(std::string const& _contract)
{
solAssert(m_assemblyStack);
2022-09-28 03:07:35 +00:00
solAssert(CompilerInputModes.count(m_options.input.mode) == 1);
std::string binary;
std::string binaryRuntime;
if (m_options.compiler.outputs.binary)
binary = objectWithLinkRefsHex(m_assemblyStack->object(_contract));
if (m_options.compiler.outputs.binaryRuntime)
binaryRuntime = objectWithLinkRefsHex(m_assemblyStack->runtimeObject(_contract));
if (m_options.compiler.outputs.binary)
{
if (!m_options.output.dir.empty())
createFile(m_assemblyStack->filesystemFriendlyName(_contract) + ".bin", binary);
else
{
sout() << "Binary:" << std::endl;
sout() << binary << std::endl;
}
}
if (m_options.compiler.outputs.binaryRuntime)
{
if (!m_options.output.dir.empty())
createFile(m_assemblyStack->filesystemFriendlyName(_contract) + ".bin-runtime", binaryRuntime);
else
{
sout() << "Binary of the runtime part:" << std::endl;
sout() << binaryRuntime << std::endl;
}
}
}
void CommandLineInterface::handleOpcode(std::string const& _contract)
{
solAssert(m_assemblyStack);
2022-09-28 03:07:35 +00:00
solAssert(CompilerInputModes.count(m_options.input.mode) == 1);
std::string opcodes{evmasm::disassemble(m_assemblyStack->object(_contract).bytecode, m_options.output.evmVersion)};
if (!m_options.output.dir.empty())
createFile(m_assemblyStack->filesystemFriendlyName(_contract) + ".opcode", opcodes);
else
{
sout() << "Opcodes:" << std::endl;
sout() << std::uppercase << opcodes;
sout() << std::endl;
}
}
void CommandLineInterface::handleIR(std::string const& _contractName)
2019-03-04 22:26:46 +00:00
{
2022-09-28 03:07:35 +00:00
solAssert(CompilerInputModes.count(m_options.input.mode) == 1);
if (!m_options.compiler.outputs.ir)
return;
if (!m_options.output.dir.empty())
createFile(m_compiler->filesystemFriendlyName(_contractName) + ".yul", m_compiler->yulIR(_contractName));
else
2019-03-04 22:26:46 +00:00
{
sout() << "IR:" << std::endl;
sout() << m_compiler->yulIR(_contractName) << std::endl;
}
}
void CommandLineInterface::handleIRAst(std::string const& _contractName)
{
solAssert(CompilerInputModes.count(m_options.input.mode) == 1);
if (!m_options.compiler.outputs.irAstJson)
return;
if (!m_options.output.dir.empty())
createFile(
m_compiler->filesystemFriendlyName(_contractName) + "_yul_ast.json",
util::jsonPrint(
m_compiler->yulIRAst(_contractName),
m_options.formatting.json
)
);
else
{
sout() << "IR AST:" << std::endl;
sout() << util::jsonPrint(
m_compiler->yulIRAst(_contractName),
m_options.formatting.json
) << std::endl;
}
}
void CommandLineInterface::handleIROptimized(std::string const& _contractName)
{
2022-09-28 03:07:35 +00:00
solAssert(CompilerInputModes.count(m_options.input.mode) == 1);
if (!m_options.compiler.outputs.irOptimized)
return;
if (!m_options.output.dir.empty())
createFile(
m_compiler->filesystemFriendlyName(_contractName) + "_opt.yul",
m_compiler->yulIROptimized(_contractName)
);
else
{
sout() << "Optimized IR:" << std::endl;
sout() << m_compiler->yulIROptimized(_contractName) << std::endl;
2019-03-04 22:26:46 +00:00
}
}
void CommandLineInterface::handleIROptimizedAst(std::string const& _contractName)
{
solAssert(CompilerInputModes.count(m_options.input.mode) == 1);
if (!m_options.compiler.outputs.irOptimizedAstJson)
return;
if (!m_options.output.dir.empty())
createFile(
m_compiler->filesystemFriendlyName(_contractName) + "_opt_yul_ast.json",
util::jsonPrint(
m_compiler->yulIROptimizedAst(_contractName),
m_options.formatting.json
)
);
else
{
sout() << "Optimized IR AST:" << std::endl;
sout() << util::jsonPrint(
m_compiler->yulIROptimizedAst(_contractName),
m_options.formatting.json
) << std::endl;
}
}
void CommandLineInterface::handleBytecode(std::string const& _contract)
{
2022-09-28 03:07:35 +00:00
solAssert(CompilerInputModes.count(m_options.input.mode) == 1);
if (m_options.compiler.outputs.opcodes)
handleOpcode(_contract);
if (m_options.compiler.outputs.binary || m_options.compiler.outputs.binaryRuntime)
handleBinary(_contract);
}
void CommandLineInterface::handleSignatureHashes(std::string const& _contract)
{
2022-09-28 03:07:35 +00:00
solAssert(CompilerInputModes.count(m_options.input.mode) == 1);
if (!m_options.compiler.outputs.signatureHashes)
return;
Json::Value interfaceSymbols = m_compiler->interfaceSymbols(_contract);
std::string out = "Function signatures:\n";
for (auto const& name: interfaceSymbols["methods"].getMemberNames())
out += interfaceSymbols["methods"][name].asString() + ": " + name + "\n";
if (interfaceSymbols.isMember("errors"))
{
out += "\nError signatures:\n";
for (auto const& name: interfaceSymbols["errors"].getMemberNames())
out += interfaceSymbols["errors"][name].asString() + ": " + name + "\n";
}
if (interfaceSymbols.isMember("events"))
{
out += "\nEvent signatures:\n";
for (auto const& name: interfaceSymbols["events"].getMemberNames())
out += interfaceSymbols["events"][name].asString() + ": " + name + "\n";
}
if (!m_options.output.dir.empty())
2017-02-05 19:19:29 +00:00
createFile(m_compiler->filesystemFriendlyName(_contract) + ".signatures", out);
else
sout() << out;
}
void CommandLineInterface::handleMetadata(std::string const& _contract)
2016-11-14 10:46:43 +00:00
{
2022-09-28 03:07:35 +00:00
solAssert(CompilerInputModes.count(m_options.input.mode) == 1);
if (!m_options.compiler.outputs.metadata)
2016-11-14 10:46:43 +00:00
return;
std::string data = m_compiler->metadata(_contract);
if (!m_options.output.dir.empty())
2017-02-05 19:19:29 +00:00
createFile(m_compiler->filesystemFriendlyName(_contract) + "_meta.json", data);
2016-11-14 10:46:43 +00:00
else
sout() << "Metadata:" << std::endl << data << std::endl;
2016-11-14 10:46:43 +00:00
}
void CommandLineInterface::handleABI(std::string const& _contract)
2017-05-06 17:02:56 +00:00
{
2022-09-28 03:07:35 +00:00
solAssert(CompilerInputModes.count(m_options.input.mode) == 1);
if (!m_options.compiler.outputs.abi)
2017-05-06 17:02:56 +00:00
return;
std::string data = jsonPrint(removeNullMembers(m_compiler->contractABI(_contract)), m_options.formatting.json);
if (!m_options.output.dir.empty())
2017-05-06 17:02:56 +00:00
createFile(m_compiler->filesystemFriendlyName(_contract) + ".abi", data);
else
sout() << "Contract JSON ABI" << std::endl << data << std::endl;
2017-05-06 17:02:56 +00:00
}
void CommandLineInterface::handleStorageLayout(std::string const& _contract)
{
2022-09-28 03:07:35 +00:00
solAssert(CompilerInputModes.count(m_options.input.mode) == 1);
if (!m_options.compiler.outputs.storageLayout)
return;
std::string data = jsonPrint(removeNullMembers(m_compiler->storageLayout(_contract)), m_options.formatting.json);
if (!m_options.output.dir.empty())
createFile(m_compiler->filesystemFriendlyName(_contract) + "_storage.json", data);
else
sout() << "Contract Storage Layout:" << std::endl << data << std::endl;
}
void CommandLineInterface::handleNatspec(bool _natspecDev, std::string const& _contract)
{
2022-09-28 03:07:35 +00:00
solAssert(CompilerInputModes.count(m_options.input.mode) == 1);
bool enabled = false;
std::string suffix;
std::string title;
2017-07-27 10:28:04 +00:00
if (_natspecDev)
{
enabled = m_options.compiler.outputs.natspecDev;
suffix = ".docdev";
title = "Developer Documentation";
2017-07-27 10:28:04 +00:00
}
else
{
enabled = m_options.compiler.outputs.natspecUser;
2017-07-27 10:28:04 +00:00
suffix = ".docuser";
title = "User Documentation";
}
if (enabled)
{
std::string output = jsonPrint(
2020-04-27 08:28:54 +00:00
removeNullMembers(
_natspecDev ?
m_compiler->natspecDev(_contract) :
m_compiler->natspecUser(_contract)
),
m_options.formatting.json
2017-07-27 10:28:04 +00:00
);
if (!m_options.output.dir.empty())
2017-02-05 19:19:29 +00:00
createFile(m_compiler->filesystemFriendlyName(_contract) + suffix, output);
else
{
sout() << title << std::endl;
sout() << output << std::endl;
}
}
}
void CommandLineInterface::handleGasEstimation(std::string const& _contract)
{
2022-09-28 03:07:35 +00:00
solAssert(CompilerInputModes.count(m_options.input.mode) == 1);
2017-04-10 13:52:13 +00:00
Json::Value estimates = m_compiler->gasEstimates(_contract);
sout() << "Gas estimation:" << std::endl;
2017-04-10 13:52:13 +00:00
if (estimates["creation"].isObject())
{
2017-04-10 13:52:13 +00:00
Json::Value creation = estimates["creation"];
sout() << "construction:" << std::endl;
sout() << " " << creation["executionCost"].asString();
sout() << " + " << creation["codeDepositCost"].asString();
sout() << " = " << creation["totalCost"].asString() << std::endl;
}
2017-04-10 13:52:13 +00:00
if (estimates["external"].isObject())
2015-05-26 09:27:59 +00:00
{
2017-04-10 13:52:13 +00:00
Json::Value externalFunctions = estimates["external"];
sout() << "external:" << std::endl;
2017-04-10 13:52:13 +00:00
for (auto const& name: externalFunctions.getMemberNames())
{
2017-04-10 13:52:13 +00:00
if (name.empty())
sout() << " fallback:\t";
2017-04-10 13:52:13 +00:00
else
sout() << " " << name << ":\t";
sout() << externalFunctions[name].asString() << std::endl;
}
2017-04-10 13:52:13 +00:00
}
if (estimates["internal"].isObject())
{
Json::Value internalFunctions = estimates["internal"];
sout() << "internal:" << std::endl;
2017-04-10 13:52:13 +00:00
for (auto const& name: internalFunctions.getMemberNames())
2015-05-26 09:27:59 +00:00
{
sout() << " " << name << ":\t";
sout() << internalFunctions[name].asString() << std::endl;
}
2015-05-26 09:27:59 +00:00
}
}
void CommandLineInterface::readInputFiles()
2016-01-25 18:42:17 +00:00
{
2022-09-28 03:07:35 +00:00
solAssert(!m_standardJsonInput.has_value());
static std::set<frontend::InputMode> const noInputFiles{
2022-09-28 03:07:35 +00:00
frontend::InputMode::Help,
frontend::InputMode::License,
frontend::InputMode::Version
};
if (noInputFiles.count(m_options.input.mode) == 1)
return;
m_fileReader.setBasePath(m_options.input.basePath);
if (m_fileReader.basePath() != "")
{
if (!boost::filesystem::exists(m_fileReader.basePath()))
solThrow(CommandLineValidationError, "Base path does not exist: \"" + m_fileReader.basePath().string() + '"');
if (!boost::filesystem::is_directory(m_fileReader.basePath()))
solThrow(CommandLineValidationError, "Base path is not a directory: \"" + m_fileReader.basePath().string() + '"');
}
2021-08-20 17:17:44 +00:00
for (boost::filesystem::path const& includePath: m_options.input.includePaths)
m_fileReader.addIncludePath(includePath);
for (boost::filesystem::path const& allowedDirectory: m_options.input.allowedDirectories)
m_fileReader.allowDirectory(allowedDirectory);
std::map<std::string, std::set<boost::filesystem::path>> collisions =
m_fileReader.detectSourceUnitNameCollisions(m_options.input.paths);
if (!collisions.empty())
{
auto pathToQuotedString = [](boost::filesystem::path const& _path){ return "\"" + _path.string() + "\""; };
std::string message =
"Source unit name collision detected. "
"The specified values of base path and/or include paths would result in multiple "
"input files being assigned the same source unit name:\n";
for (auto const& [sourceUnitName, normalizedInputPaths]: collisions)
{
message += sourceUnitName + " matches: ";
message += util::joinHumanReadable(normalizedInputPaths | ranges::views::transform(pathToQuotedString)) + "\n";
}
solThrow(CommandLineValidationError, message);
}
for (boost::filesystem::path const& infile: m_options.input.paths)
{
if (!boost::filesystem::exists(infile))
2016-01-25 18:42:17 +00:00
{
if (!m_options.input.ignoreMissingFiles)
solThrow(CommandLineValidationError, '"' + infile.string() + "\" is not found.");
2016-01-25 18:42:17 +00:00
else
serr() << infile << " is not found. Skipping." << std::endl;
2016-01-25 18:42:17 +00:00
continue;
}
2016-01-25 18:42:17 +00:00
if (!boost::filesystem::is_regular_file(infile))
{
if (!m_options.input.ignoreMissingFiles)
solThrow(CommandLineValidationError, '"' + infile.string() + "\" is not a valid file.");
else
serr() << infile << " is not a valid file. Skipping." << std::endl;
continue;
2016-01-25 18:42:17 +00:00
}
// NOTE: we ignore the FileNotFound exception as we manually check above
std::string fileContent = readFileAsString(infile);
if (m_options.input.mode == InputMode::StandardJson)
{
2022-09-28 03:07:35 +00:00
solAssert(!m_standardJsonInput.has_value());
2022-09-01 08:55:29 +00:00
m_standardJsonInput = std::move(fileContent);
}
else
{
2022-09-01 08:55:29 +00:00
m_fileReader.addOrUpdateFile(infile, std::move(fileContent));
m_fileReader.allowDirectory(boost::filesystem::canonical(infile).remove_filename());
}
}
if (m_options.input.addStdin)
{
if (m_options.input.mode == InputMode::StandardJson)
{
2022-09-28 03:07:35 +00:00
solAssert(!m_standardJsonInput.has_value());
m_standardJsonInput = readUntilEnd(m_sin);
}
else
m_fileReader.setStdin(readUntilEnd(m_sin));
}
if (
m_options.input.mode != InputMode::LanguageServer &&
m_fileReader.sourceUnits().empty() &&
!m_standardJsonInput.has_value()
)
solThrow(CommandLineValidationError, "All specified input files either do not exist or are not regular files.");
2016-01-25 18:42:17 +00:00
}
std::map<std::string, Json::Value> CommandLineInterface::parseAstFromInput()
2019-09-11 19:16:35 +00:00
{
2022-09-28 03:07:35 +00:00
solAssert(m_options.input.mode == InputMode::CompilerWithASTImport);
std::map<std::string, Json::Value> sourceJsons;
std::map<std::string, std::string> tmpSources;
2019-09-11 19:16:35 +00:00
2021-12-13 13:52:17 +00:00
for (SourceCode const& sourceCode: m_fileReader.sourceUnits() | ranges::views::values)
2019-09-11 19:16:35 +00:00
{
Json::Value ast;
astAssert(jsonParseStrict(sourceCode, ast), "Input file could not be parsed to JSON");
2019-09-11 19:16:35 +00:00
astAssert(ast.isMember("sources"), "Invalid Format for import-JSON: Must have 'sources'-object");
for (auto& src: ast["sources"].getMemberNames())
{
std::string astKey = ast["sources"][src].isMember("ast") ? "ast" : "AST";
2019-09-11 19:16:35 +00:00
astAssert(ast["sources"][src].isMember(astKey), "astkey is not member");
astAssert(ast["sources"][src][astKey]["nodeType"].asString() == "SourceUnit", "Top-level node should be a 'SourceUnit'");
astAssert(sourceJsons.count(src) == 0, "All sources must have unique names");
2022-09-01 08:55:29 +00:00
sourceJsons.emplace(src, std::move(ast["sources"][src][astKey]));
2019-09-11 19:16:35 +00:00
tmpSources[src] = util::jsonCompactPrint(ast);
}
}
2021-12-13 13:52:17 +00:00
m_fileReader.setSourceUnits(tmpSources);
2019-09-11 19:16:35 +00:00
return sourceJsons;
}
void CommandLineInterface::createFile(std::string const& _fileName, std::string const& _data)
{
namespace fs = boost::filesystem;
2022-09-28 03:07:35 +00:00
solAssert(!m_options.output.dir.empty());
// NOTE: create_directories() raises an exception if the path consists solely of '.' or '..'
// (or equivalent such as './././.'). Paths like 'a/b/.' and 'a/b/..' are fine though.
// The simplest workaround is to use an absolute path.
fs::create_directories(fs::absolute(m_options.output.dir));
std::string pathName = (m_options.output.dir / _fileName).string();
if (fs::exists(pathName) && !m_options.output.overwriteFiles)
solThrow(CommandLineOutputError, "Refusing to overwrite existing file \"" + pathName + "\" (use --overwrite to force).");
std::ofstream outFile(pathName);
2015-07-21 13:29:15 +00:00
outFile << _data;
if (!outFile)
solThrow(CommandLineOutputError, "Could not write to file \"" + pathName + "\".");
}
void CommandLineInterface::createJson(std::string const& _fileName, std::string const& _json)
{
createFile(boost::filesystem::path(_fileName).stem().string() + std::string(".json"), _json);
}
bool CommandLineInterface::run(int _argc, char const* const* _argv)
{
try
{
if (!parseArguments(_argc, _argv))
return false;
readInputFiles();
processInput();
return true;
}
catch (CommandLineError const& _exception)
{
m_hasOutput = true;
// There might be no message in the exception itself if the error output is bulky and has
// already been printed to stderr (this happens e.g. for compiler errors).
if (_exception.what() != ""s)
serr() << _exception.what() << std::endl;
return false;
}
}
bool CommandLineInterface::parseArguments(int _argc, char const* const* _argv)
{
CommandLineParser parser;
if (isatty(fileno(stdin)) && _argc == 1)
{
// If the terminal is taking input from the user, provide more user-friendly output.
CommandLineParser::printHelp(sout());
// In this case we want to exit with an error but not display any error message.
return false;
}
parser.parse(_argc, _argv);
m_options = parser.options();
return true;
}
void CommandLineInterface::processInput()
{
switch (m_options.input.mode)
2017-04-19 15:59:03 +00:00
{
case InputMode::Help:
CommandLineParser::printHelp(sout());
2021-10-11 11:30:08 +00:00
break;
case InputMode::License:
printLicense();
break;
case InputMode::Version:
printVersion();
break;
case InputMode::StandardJson:
2017-02-09 14:55:57 +00:00
{
2022-09-28 03:07:35 +00:00
solAssert(m_standardJsonInput.has_value());
StandardCompiler compiler(m_universalCallback.callback(), m_options.formatting.json);
sout() << compiler.compile(std::move(m_standardJsonInput.value())) << std::endl;
m_standardJsonInput.reset();
break;
2017-02-09 14:55:57 +00:00
}
case InputMode::LanguageServer:
serveLSP();
break;
case InputMode::Assembler:
assembleYul(m_options.assembly.inputLanguage, m_options.assembly.targetMachine);
break;
case InputMode::Linker:
link();
writeLinkedFiles();
break;
case InputMode::Compiler:
case InputMode::CompilerWithASTImport:
compile();
outputCompilationResults();
break;
case InputMode::EVMAssemblerJSON:
assembleFromEvmAssemblyJson();
handleCombinedJSON();
handleBytecode(m_assemblyStack->contractNames().front());
handleEVMAssembly(m_assemblyStack->contractNames().front());
break;
}
}
void CommandLineInterface::printVersion()
{
sout() << "solc, the solidity compiler commandline interface" << std::endl;
sout() << "Version: " << solidity::frontend::VersionString << std::endl;
}
void CommandLineInterface::printLicense()
{
sout() << otherLicenses << std::endl;
// This is a static variable generated by cmake from LICENSE.txt
sout() << licenseText << std::endl;
}
void CommandLineInterface::assembleFromEvmAssemblyJson()
{
solAssert(m_options.input.mode == InputMode::EVMAssemblerJSON);
solAssert(!m_assemblyStack);
solAssert(!m_evmAssemblyStack && !m_compiler);
std::unique_ptr<evmasm::EVMAssemblyStack> evmAssemblyStack;
solAssert(m_fileReader.sourceUnits().size() == 1);
auto&& [sourceUnitName, source] = *m_fileReader.sourceUnits().begin();
try
{
evmAssemblyStack = std::make_unique<evmasm::EVMAssemblyStack>(m_options.output.evmVersion);
evmAssemblyStack->parseAndAnalyze(sourceUnitName, source);
if (m_options.output.debugInfoSelection.has_value())
evmAssemblyStack->selectDebugInfo(m_options.output.debugInfoSelection.value());
evmAssemblyStack->assemble();
m_evmAssemblyStack = std::move(evmAssemblyStack);
m_assemblyStack = m_evmAssemblyStack.get();
}
catch (evmasm::AssemblyImportException const& _exception)
{
solThrow(CommandLineExecutionError, "Assembly Import Error: "s + _exception.what());
}
}
void CommandLineInterface::compile()
{
2022-09-28 03:07:35 +00:00
solAssert(CompilerInputModes.count(m_options.input.mode) == 1);
solAssert(!m_assemblyStack);
solAssert(!m_evmAssemblyStack && !m_compiler);
2020-11-02 20:20:20 +00:00
m_compiler = std::make_unique<CompilerStack>(m_universalCallback.callback());
m_assemblyStack = m_compiler.get();
2021-06-29 12:38:59 +00:00
SourceReferenceFormatter formatter(serr(false), *m_compiler, coloredOutput(m_options), m_options.formatting.withErrorIds);
try
{
if (m_options.metadata.literalSources)
m_compiler->useMetadataLiteralSources(true);
m_compiler->setMetadataFormat(m_options.metadata.format);
m_compiler->setMetadataHash(m_options.metadata.hash);
if (m_options.modelChecker.initialize)
m_compiler->setModelCheckerSettings(m_options.modelChecker.settings);
m_compiler->setRemappings(m_options.input.remappings);
m_compiler->setLibraries(m_options.linker.libraries);
m_compiler->setViaIR(m_options.output.viaIR);
m_compiler->setEVMVersion(m_options.output.evmVersion);
m_compiler->setEOFVersion(m_options.output.eofVersion);
m_compiler->setRevertStringBehaviour(m_options.output.revertStrings);
if (m_options.output.debugInfoSelection.has_value())
m_compiler->selectDebugInfo(m_options.output.debugInfoSelection.value());
// TODO: Perhaps we should not compile unless requested
m_compiler->enableIRGeneration(
m_options.compiler.outputs.ir ||
m_options.compiler.outputs.irOptimized ||
m_options.compiler.outputs.irAstJson ||
m_options.compiler.outputs.irOptimizedAstJson
);
m_compiler->enableEvmBytecodeGeneration(
m_options.compiler.estimateGas ||
m_options.compiler.outputs.asm_ ||
m_options.compiler.outputs.asmJson ||
m_options.compiler.outputs.opcodes ||
m_options.compiler.outputs.binary ||
m_options.compiler.outputs.binaryRuntime ||
(m_options.compiler.combinedJsonRequests && (
m_options.compiler.combinedJsonRequests->binary ||
m_options.compiler.combinedJsonRequests->binaryRuntime ||
m_options.compiler.combinedJsonRequests->opcodes ||
m_options.compiler.combinedJsonRequests->asm_ ||
m_options.compiler.combinedJsonRequests->generatedSources ||
m_options.compiler.combinedJsonRequests->generatedSourcesRuntime ||
m_options.compiler.combinedJsonRequests->srcMap ||
m_options.compiler.combinedJsonRequests->srcMapRuntime ||
m_options.compiler.combinedJsonRequests->funDebug ||
m_options.compiler.combinedJsonRequests->funDebugRuntime
))
);
2019-03-04 22:26:46 +00:00
m_compiler->setOptimiserSettings(m_options.optimiserSettings());
if (m_options.input.mode == InputMode::CompilerWithASTImport)
2019-09-11 19:16:35 +00:00
{
try
{
m_compiler->importASTs(parseAstFromInput());
if (!m_compiler->analyze())
{
2021-06-29 12:38:59 +00:00
formatter.printErrorInformation(m_compiler->errors());
2019-09-11 19:16:35 +00:00
astAssert(false, "Analysis of the AST failed");
}
}
catch (Exception const& _exc)
{
// FIXME: AST import is missing proper validations. This hack catches failing
// assertions and presents them as if they were compiler errors.
solThrow(CommandLineExecutionError, "Failed to import AST: "s + _exc.what());
2019-09-11 19:16:35 +00:00
}
}
else
2021-12-13 13:52:17 +00:00
m_compiler->setSources(m_fileReader.sourceUnits());
2019-09-11 19:16:35 +00:00
bool successful = m_compiler->compile(m_options.output.stopAfter);
2015-10-21 14:43:31 +00:00
for (auto const& error: m_compiler->errors())
{
m_hasOutput = true;
2020-10-30 21:11:29 +00:00
formatter.printErrorInformation(*error);
}
2015-10-21 14:43:31 +00:00
if (!successful)
solThrow(CommandLineExecutionError, "");
}
catch (CompilerError const& _exception)
{
m_hasOutput = true;
formatter.printExceptionInformation(
_exception,
Error::errorSeverity(Error::Type::CompilerError)
);
solThrow(CommandLineExecutionError, "");
}
catch (Error const& _error)
{
if (_error.type() == Error::Type::DocstringParsingError)
{
serr() << *boost::get_error_info<errinfo_comment>(_error);
solThrow(CommandLineExecutionError, "Documentation parsing failed.");
}
else
{
m_hasOutput = true;
formatter.printErrorInformation(_error);
solThrow(CommandLineExecutionError, "");
}
}
}
void CommandLineInterface::handleCombinedJSON()
{
solAssert(m_assemblyStack);
2022-09-28 03:07:35 +00:00
solAssert(CompilerInputModes.count(m_options.input.mode) == 1);
if (!m_options.compiler.combinedJsonRequests.has_value())
return;
Json::Value output(Json::objectValue);
2019-12-11 16:31:36 +00:00
output[g_strVersion] = frontend::VersionString;
std::vector<std::string> contracts = m_assemblyStack->contractNames();
if (!contracts.empty())
output[g_strContracts] = Json::Value(Json::objectValue);
for (std::string const& contractName: contracts)
{
2017-07-27 14:40:01 +00:00
Json::Value& contractData = output[g_strContracts][contractName] = Json::objectValue;
if (m_options.compiler.combinedJsonRequests->abi && m_compiler->compilationSuccessful())
contractData[g_strAbi] = m_compiler->contractABI(contractName);
if (m_options.compiler.combinedJsonRequests->metadata && m_compiler->compilationSuccessful())
2017-05-19 15:10:32 +00:00
contractData["metadata"] = m_compiler->metadata(contractName);
if (m_options.compiler.combinedJsonRequests->binary && m_assemblyStack->compilationSuccessful())
contractData[g_strBinary] = m_assemblyStack->object(contractName).toHex();
if (m_options.compiler.combinedJsonRequests->binaryRuntime && m_assemblyStack->compilationSuccessful())
contractData[g_strBinaryRuntime] = m_assemblyStack->runtimeObject(contractName).toHex();
if (m_options.compiler.combinedJsonRequests->opcodes && m_assemblyStack->compilationSuccessful())
contractData[g_strOpcodes] = evmasm::disassemble(m_assemblyStack->object(contractName).bytecode, m_options.output.evmVersion);
if (m_options.compiler.combinedJsonRequests->asm_ && m_assemblyStack->compilationSuccessful())
contractData[g_strAsm] = m_assemblyStack->assemblyJSON(contractName);
if (m_options.compiler.combinedJsonRequests->storageLayout && m_compiler->compilationSuccessful())
contractData[g_strStorageLayout] = m_compiler->storageLayout(contractName);
if (m_options.compiler.combinedJsonRequests->generatedSources && m_compiler->compilationSuccessful())
contractData[g_strGeneratedSources] = m_compiler->generatedSources(contractName, false);
if (m_options.compiler.combinedJsonRequests->generatedSourcesRuntime && m_compiler->compilationSuccessful())
contractData[g_strGeneratedSourcesRuntime] = m_compiler->generatedSources(contractName, true);
if (m_options.compiler.combinedJsonRequests->srcMap && m_assemblyStack->compilationSuccessful())
2016-07-01 08:14:50 +00:00
{
auto map = m_assemblyStack->sourceMapping(contractName);
contractData[g_strSrcMap] = map ? *map : "";
2016-07-01 08:14:50 +00:00
}
if (m_options.compiler.combinedJsonRequests->srcMapRuntime && m_assemblyStack->compilationSuccessful())
2016-07-01 08:14:50 +00:00
{
auto map = m_assemblyStack->runtimeSourceMapping(contractName);
contractData[g_strSrcMapRuntime] = map ? *map : "";
2016-07-01 08:14:50 +00:00
}
if (m_options.compiler.combinedJsonRequests->funDebug && m_assemblyStack->compilationSuccessful())
2020-05-07 12:46:47 +00:00
contractData[g_strFunDebug] = StandardCompiler::formatFunctionDebugData(
m_assemblyStack->object(contractName).functionDebugData
2020-05-07 12:46:47 +00:00
);
if (m_options.compiler.combinedJsonRequests->funDebugRuntime && m_assemblyStack->compilationSuccessful())
2020-05-07 12:46:47 +00:00
contractData[g_strFunDebugRuntime] = StandardCompiler::formatFunctionDebugData(
m_assemblyStack->runtimeObject(contractName).functionDebugData
2020-05-07 12:46:47 +00:00
);
if (m_options.compiler.combinedJsonRequests->signatureHashes && m_compiler->compilationSuccessful())
contractData[g_strSignatureHashes] = m_compiler->interfaceSymbols(contractName)["methods"];
if (m_options.compiler.combinedJsonRequests->natspecDev && m_compiler->compilationSuccessful())
contractData[g_strNatspecDev] = m_compiler->natspecDev(contractName);
if (m_options.compiler.combinedJsonRequests->natspecUser && m_compiler->compilationSuccessful())
contractData[g_strNatspecUser] = m_compiler->natspecUser(contractName);
}
bool needsSourceList = m_options.compiler.combinedJsonRequests->ast ||
m_options.compiler.combinedJsonRequests->srcMap ||
m_options.compiler.combinedJsonRequests->srcMapRuntime;
2016-07-01 08:14:50 +00:00
if (needsSourceList)
{
// Indices into this array are used to abbreviate source names in source locations.
output[g_strSourceList] = Json::Value(Json::arrayValue);
2016-07-01 08:14:50 +00:00
for (auto const& source: m_assemblyStack->sourceNames())
output[g_strSourceList].append(source);
2016-07-01 08:14:50 +00:00
}
if (m_options.compiler.combinedJsonRequests->ast)
{
output[g_strSources] = Json::Value(Json::objectValue);
2021-12-13 13:52:17 +00:00
for (auto const& sourceCode: m_fileReader.sourceUnits())
{
output[g_strSources][sourceCode.first] = Json::Value(Json::objectValue);
output[g_strSources][sourceCode.first]["AST"] = ASTJsonExporter(
m_compiler->state(),
m_compiler->sourceIndices()
).toJson(m_compiler->ast(sourceCode.first));
output[g_strSources][sourceCode.first]["id"] = m_compiler->sourceIndices().at(sourceCode.first);
}
}
std::string json = jsonPrint(removeNullMembers(std::move(output)), m_options.formatting.json);
if (!m_options.output.dir.empty())
2017-07-03 23:46:30 +00:00
createJson("combined", json);
else
sout() << json << std::endl;
}
2020-11-20 14:35:53 +00:00
void CommandLineInterface::handleAst()
{
2022-09-28 03:07:35 +00:00
solAssert(CompilerInputModes.count(m_options.input.mode) == 1);
if (!m_options.compiler.outputs.astCompactJson)
2020-11-20 14:35:53 +00:00
return;
std::vector<ASTNode const*> asts;
2021-12-13 13:52:17 +00:00
for (auto const& sourceCode: m_fileReader.sourceUnits())
2020-11-20 14:35:53 +00:00
asts.push_back(&m_compiler->ast(sourceCode.first));
if (!m_options.output.dir.empty())
{
2021-12-13 13:52:17 +00:00
for (auto const& sourceCode: m_fileReader.sourceUnits())
{
std::stringstream data;
std::string postfix = "";
ASTJsonExporter(m_compiler->state(), m_compiler->sourceIndices()).print(data, m_compiler->ast(sourceCode.first), m_options.formatting.json);
2020-11-20 14:35:53 +00:00
postfix += "_json";
boost::filesystem::path path(sourceCode.first);
createFile(path.filename().string() + postfix + ".ast", data.str());
}
2020-11-20 14:35:53 +00:00
}
else
{
sout() << "JSON AST (compact format):" << std::endl << std::endl;
2021-12-13 13:52:17 +00:00
for (auto const& sourceCode: m_fileReader.sourceUnits())
{
sout() << std::endl << "======= " << sourceCode.first << " =======" << std::endl;
ASTJsonExporter(m_compiler->state(), m_compiler->sourceIndices()).print(sout(), m_compiler->ast(sourceCode.first), m_options.formatting.json);
}
}
}
void CommandLineInterface::serveLSP()
{
lsp::StdioTransport transport;
if (!lsp::LanguageServer{transport}.run())
solThrow(CommandLineExecutionError, "LSP terminated abnormally.");
}
void CommandLineInterface::link()
2015-09-11 17:35:01 +00:00
{
2022-09-28 03:07:35 +00:00
solAssert(m_options.input.mode == InputMode::Linker);
2016-09-06 09:12:55 +00:00
// Map from how the libraries will be named inside the bytecode to their addresses.
std::map<std::string, h160> librariesReplacements;
2016-09-06 09:57:21 +00:00
int const placeholderSize = 40; // 20 bytes or 40 hex characters
for (auto const& library: m_options.linker.libraries)
{
std::string const& name = library.first;
2016-09-06 09:12:55 +00:00
// Library placeholders are 40 hex digits (20 bytes) that start and end with '__'.
2018-10-04 12:55:02 +00:00
// This leaves 36 characters for the library identifier. The identifier used to
// be just the cropped or '_'-padded library name, but this changed to
// the cropped hex representation of the hash of the library name.
// We support both ways of linking here.
2019-12-11 16:31:36 +00:00
librariesReplacements["__" + evmasm::LinkerObject::libraryPlaceholder(name) + "__"] = library.second;
2018-10-04 12:55:02 +00:00
std::string replacement = "__";
2016-09-06 09:12:55 +00:00
for (size_t i = 0; i < placeholderSize - 4; ++i)
replacement.push_back(i < name.size() ? name[i] : '_');
replacement += "__";
librariesReplacements[replacement] = library.second;
}
2021-12-13 13:52:17 +00:00
FileReader::StringMap sourceCodes = m_fileReader.sourceUnits();
for (auto& src: sourceCodes)
2015-09-11 17:35:01 +00:00
{
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 ||
*(it + 1) != '_' ||
*(it + placeholderSize - 2) != '_' ||
*(it + placeholderSize - 1) != '_'
)
solThrow(
CommandLineExecutionError,
"Error in binary object file " + src.first + " at position " + std::to_string(it - src.second.begin()) + "\n" +
'"' + std::string(it, it + std::min(placeholderSize, static_cast<int>(end - it))) + "\" is not a valid link reference."
);
2015-09-11 17:35:01 +00:00
std::string foundPlaceholder(it, it + placeholderSize);
if (librariesReplacements.count(foundPlaceholder))
2015-09-11 17:35:01 +00:00
{
std::string hexStr(util::toHex(librariesReplacements.at(foundPlaceholder).asBytes()));
copy(hexStr.begin(), hexStr.end(), it);
2015-09-11 17:35:01 +00:00
}
else
serr() << "Reference \"" << foundPlaceholder << "\" in file \"" << src.first << "\" still unresolved." << std::endl;
2016-09-06 09:12:55 +00:00
it += placeholderSize;
2015-09-11 17:35:01 +00:00
}
2018-10-04 12:55:02 +00:00
// Remove hints for resolved libraries.
for (auto const& library: m_options.linker.libraries)
2018-10-04 12:55:02 +00:00
boost::algorithm::erase_all(src.second, "\n" + libraryPlaceholderHint(library.first));
while (!src.second.empty() && *prev(src.second.end()) == '\n')
2019-02-13 15:56:46 +00:00
src.second.resize(src.second.size() - 1);
2015-09-11 17:35:01 +00:00
}
2022-09-01 08:55:29 +00:00
m_fileReader.setSourceUnits(std::move(sourceCodes));
2015-09-11 17:35:01 +00:00
}
void CommandLineInterface::writeLinkedFiles()
{
2022-09-28 03:07:35 +00:00
solAssert(m_options.input.mode == InputMode::Linker);
2021-12-13 13:52:17 +00:00
for (auto const& src: m_fileReader.sourceUnits())
2015-09-11 17:35:01 +00:00
if (src.first == g_stdinFileName)
sout() << src.second << std::endl;
2015-09-11 17:35:01 +00:00
else
{
std::ofstream outFile(src.first);
outFile << src.second;
if (!outFile)
solThrow(CommandLineOutputError, "Could not write to file " + src.first + ". Aborting.");
}
sout() << "Linking completed." << std::endl;
2015-09-11 17:35:01 +00:00
}
std::string CommandLineInterface::libraryPlaceholderHint(std::string const& _libraryName)
2018-10-04 12:55:02 +00:00
{
2019-12-11 16:31:36 +00:00
return "// " + evmasm::LinkerObject::libraryPlaceholder(_libraryName) + " -> " + _libraryName;
2018-10-04 12:55:02 +00:00
}
std::string CommandLineInterface::objectWithLinkRefsHex(evmasm::LinkerObject const& _obj)
2018-10-04 12:55:02 +00:00
{
std::string out = _obj.toHex();
2018-10-04 12:55:02 +00:00
if (!_obj.linkReferences.empty())
{
out += "\n";
for (auto const& linkRef: _obj.linkReferences)
out += "\n" + libraryPlaceholderHint(linkRef.second);
}
return out;
}
void CommandLineInterface::assembleYul(yul::YulStack::Language _language, yul::YulStack::Machine _targetMachine)
2016-02-22 01:13:41 +00:00
{
2022-09-28 03:07:35 +00:00
solAssert(m_options.input.mode == InputMode::Assembler);
2016-02-22 01:13:41 +00:00
bool successful = true;
std::map<std::string, yul::YulStack> yulStacks;
2021-12-13 13:52:17 +00:00
for (auto const& src: m_fileReader.sourceUnits())
2016-02-22 01:13:41 +00:00
{
auto& stack = yulStacks[src.first] = yul::YulStack(
m_options.output.evmVersion,
m_options.output.eofVersion,
_language,
m_options.optimiserSettings(),
m_options.output.debugInfoSelection.has_value() ?
m_options.output.debugInfoSelection.value() :
DebugInfoSelection::Default()
);
if (!stack.parseAndAnalyze(src.first, src.second))
successful = false;
else
stack.optimize();
2016-02-22 01:13:41 +00:00
}
for (auto const& sourceAndStack: yulStacks)
{
auto const& stack = sourceAndStack.second;
2021-06-29 12:38:59 +00:00
SourceReferenceFormatter formatter(serr(false), stack, coloredOutput(m_options), m_options.formatting.withErrorIds);
for (auto const& error: stack.errors())
{
m_hasOutput = true;
2020-10-30 21:11:29 +00:00
formatter.printErrorInformation(*error);
}
2021-06-30 12:48:45 +00:00
if (Error::containsErrors(stack.errors()))
successful = false;
}
2016-02-22 01:13:41 +00:00
if (!successful)
{
solAssert(m_hasOutput);
solThrow(CommandLineExecutionError, "");
}
2016-02-22 01:13:41 +00:00
2021-12-13 13:52:17 +00:00
for (auto const& src: m_fileReader.sourceUnits())
{
2023-02-22 15:32:29 +00:00
solAssert(_targetMachine == yul::YulStack::Machine::EVM);
std::string machine = "EVM";
sout() << std::endl << "======= " << src.first << " (" << machine << ") =======" << std::endl;
yul::YulStack& stack = yulStacks[src.first];
2017-05-29 23:12:38 +00:00
if (m_options.compiler.outputs.irOptimized)
{
// NOTE: This actually outputs unoptimized code when the optimizer is disabled but
// 'ir' output in StandardCompiler works the same way.
sout() << std::endl << "Pretty printed source:" << std::endl;
sout() << stack.print() << std::endl;
}
2017-05-29 23:12:38 +00:00
2019-02-13 11:07:20 +00:00
yul::MachineAssemblyObject object;
object = stack.assemble(_targetMachine);
object.bytecode->link(m_options.linker.libraries);
2017-05-29 23:12:38 +00:00
if (m_options.compiler.outputs.binary)
{
sout() << std::endl << "Binary representation:" << std::endl;
if (object.bytecode)
sout() << object.bytecode->toHex() << std::endl;
else
serr() << "No binary representation found." << std::endl;
}
if (m_options.compiler.outputs.astCompactJson)
{
sout() << "AST:" << std::endl << std::endl;
sout() << util::jsonPrint(stack.astJson(), m_options.formatting.json) << std::endl;
}
2023-02-22 15:32:29 +00:00
solAssert(_targetMachine == yul::YulStack::Machine::EVM, "");
if (m_options.compiler.outputs.asm_)
{
sout() << std::endl << "Text representation:" << std::endl;
if (!object.assembly.empty())
sout() << object.assembly << std::endl;
else
serr() << "No text representation found." << std::endl;
}
if (m_options.compiler.outputs.asmJson)
{
std::shared_ptr<evmasm::Assembly> assembly{stack.assembleEVMWithDeployed().first};
if (assembly)
{
std::function<std::map<std::string, unsigned>(yul::Object const&)> collectSourceIndices =
[&](yul::Object const& _object) -> std::map<std::string, unsigned> {
std::map<std::string, unsigned> sourceIndices;
if (_object.debugData && _object.debugData->sourceNames.has_value())
for (auto const& iter: *_object.debugData->sourceNames)
sourceIndices[*iter.second] = iter.first;
for (auto const& sub: _object.subObjects)
{
auto subObject = dynamic_cast<yul::Object const*>(sub.get());
if (subObject)
for (auto const& [name, index]: collectSourceIndices(*subObject))
sourceIndices[name] = index;
}
return sourceIndices;
};
if (stack.parserResult() && stack.parserResult()->debugData)
{
std::map<std::string, unsigned> sourceIndices = collectSourceIndices(*stack.parserResult());
// if sourceIndices are empty here, there were no source locations annotated in the yul source.
// in this case, we just add the filename of the yul file itself here.
if (sourceIndices.empty())
sourceIndices[src.first] = 0;
size_t max_index = 0;
for (auto const& [name, index]: sourceIndices)
if (max_index < index)
max_index = index;
std::vector<std::string> sourceList(max_index + 1);
uint32_t counter{0};
for (auto& source: sourceList)
source = "unknown-source-" + std::to_string(counter++);
for (auto const& [name, index]: sourceIndices)
sourceList[index] = name;
sout() << util::jsonPrint(
removeNullMembers(
assembly->assemblyJSON(sourceList)
),
m_options.formatting.json
) << std::endl;
return;
}
}
serr() << "Could not create Assembly JSON representation." << std::endl;
}
}
}
2015-09-11 17:35:01 +00:00
void CommandLineInterface::outputCompilationResults()
{
2022-09-28 03:07:35 +00:00
solAssert(CompilerInputModes.count(m_options.input.mode) == 1);
handleCombinedJSON();
// do we need AST output?
2020-11-20 14:35:53 +00:00
handleAst();
CompilerOutputs astOutputSelection;
astOutputSelection.astCompactJson = true;
if (m_options.compiler.outputs != CompilerOutputs() && m_options.compiler.outputs != astOutputSelection)
{
// Currently AST is the only output allowed with --stop-after parsing. For all of the others
// we can safely assume that full compilation was performed and successful.
solAssert(m_options.output.stopAfter >= CompilerStack::State::CompilationSuccessful);
for (std::string const& contract: m_compiler->contractNames())
{
if (needsHumanTargetedStdout(m_options))
sout() << std::endl << "======= " << contract << " =======" << std::endl;
2017-09-11 14:18:56 +00:00
handleEVMAssembly(contract);
if (m_options.compiler.estimateGas)
handleGasEstimation(contract);
handleBytecode(contract);
handleIR(contract);
handleIRAst(contract);
handleIROptimized(contract);
handleIROptimizedAst(contract);
handleSignatureHashes(contract);
handleMetadata(contract);
handleABI(contract);
handleStorageLayout(contract);
handleNatspec(true, contract);
handleNatspec(false, contract);
} // end of contracts iteration
}
if (!m_hasOutput)
{
if (!m_options.output.dir.empty())
sout() << "Compiler run successful. Artifact(s) can be found in directory " << m_options.output.dir << "." << std::endl;
else if (m_compiler->contractNames().empty())
sout() << "Compiler run successful. No contracts to compile." << std::endl;
else
sout() << "Compiler run successful. No output generated." << std::endl;
}
}
}