diff --git a/libsolidity/Version.cpp b/libsolidity/Version.cpp index 75a0dd256..61b2e95ad 100644 --- a/libsolidity/Version.cpp +++ b/libsolidity/Version.cpp @@ -29,7 +29,7 @@ using namespace dev; using namespace dev::solidity; using namespace std; -char const* dev::solidity::VersionNumber = "0.1.1"; +char const* dev::solidity::VersionNumber = "0.1.2"; extern string const dev::solidity::VersionString = string(dev::solidity::VersionNumber) + "-" + diff --git a/solc/CommandLineInterface.cpp b/solc/CommandLineInterface.cpp index a5ad07d7c..8c8ff03ea 100644 --- a/solc/CommandLineInterface.cpp +++ b/solc/CommandLineInterface.cpp @@ -121,48 +121,29 @@ void CommandLineInterface::handleBinary(string const& _contract) if (m_args.count(g_argBinaryStr)) { if (m_args.count("output-dir")) - { - stringstream data; - data << toHex(m_compiler->getBytecode(_contract)); - createFile(_contract + ".bin", data.str()); - } + createFile(_contract + ".bin", toHex(m_compiler->getBytecode(_contract))); else { cout << "Binary: " << endl; cout << toHex(m_compiler->getBytecode(_contract)) << endl; } - } if (m_args.count(g_argCloneBinaryStr)) { if (m_args.count("output-dir")) - { - stringstream data; - data << toHex(m_compiler->getCloneBytecode(_contract)); - createFile(_contract + ".clone_bin", data.str()); - } + createFile(_contract + ".clone_bin", toHex(m_compiler->getCloneBytecode(_contract))); else { cout << "Clone Binary: " << endl; cout << toHex(m_compiler->getCloneBytecode(_contract)) << endl; } } - else - { - cout << "Binary: " << endl; - cout << toHex(m_compiler->getBytecode(_contract)) << endl; - } - } void CommandLineInterface::handleOpcode(string const& _contract) { if (m_args.count("output-dir")) - { - stringstream data; - data << eth::disassemble(m_compiler->getBytecode(_contract)); - createFile(_contract + ".opcode", data.str()); - } + createFile(_contract + ".opcode", eth::disassemble(m_compiler->getBytecode(_contract))); else { cout << "Opcodes: " << endl; @@ -190,11 +171,7 @@ void CommandLineInterface::handleSignatureHashes(string const& _contract) out += toHex(it.first.ref()) + ": " + it.second->externalSignature() + "\n"; if (m_args.count("output-dir")) - { - stringstream data; - data << out; - createFile(_contract + ".signatures", data.str()); - } + createFile(_contract + ".signatures", out); else cout << "Function signatures: " << endl << out; } @@ -234,11 +211,7 @@ void CommandLineInterface::handleMeta(DocumentationType _type, string const& _co if (m_args.count(argName)) { if (m_args.count("output-dir")) - { - stringstream data; - data << m_compiler->getMetadata(_contract, _type); - createFile(_contract + suffix, data.str()); - } + createFile(_contract + suffix, m_compiler->getMetadata(_contract, _type)); else { cout << title << endl; @@ -301,55 +274,79 @@ void CommandLineInterface::createFile(string const& _fileName, string const& _da { namespace fs = boost::filesystem; // create directory if not existent - fs::path p(m_args["output-dir"].as()); + fs::path p(m_args.at("output-dir").as()); fs::create_directories(p); - ofstream outFile((p / _fileName).string()); + string pathName = (p / _fileName).string(); + ofstream outFile(pathName); outFile << _data; if (!outFile) - BOOST_THROW_EXCEPTION(FileError() << errinfo_comment("Could not write to file: " + _fileName)); + BOOST_THROW_EXCEPTION(FileError() << errinfo_comment("Could not write to file: " + pathName)); } bool CommandLineInterface::parseArguments(int _argc, char** _argv) { // Declare the supported options. - po::options_description desc("Allowed options"); + po::options_description desc( + R"(solc, the Solidity commandline compiler. +Usage: solc [options] [input_file...] +Compiles the given Solidity input files (or the standard input if none given) and +outputs the components specified in the options at standard output or in files in +the output directory, if specified. +Example: solc --bin -o /tmp/solcoutput contract.sol + +Allowed options)", + po::options_description::m_default_line_length, + po::options_description::m_default_line_length - 23); desc.add_options() - ("help", "Show help message and exit") - ("version", "Show version and exit") - ("optimize", po::value()->default_value(false), "Optimize bytecode") - ("optimize-runs", po::value()->default_value(200), "Estimated number of contract runs for optimizer.") - ("add-std", po::value()->default_value(false), "Add standard contracts") - ("input-file", po::value>(), "input file") - ("output-dir,o", po::value(), "Output directory path") + ("help", "Show help message and exit.") + ("version", "Show version and exit.") + ("optimize", "Enable bytecode optimizer.") + ( + "optimize-runs", + po::value()->value_name("n")->default_value(200), + "Estimated number of contract runs for optimizer tuning." + ) + (g_argAddStandard.c_str(), "Add standard contracts.") + ( + "output-dir,o", + po::value()->value_name("path"), + "If given, creates one file per component and contract/file at the specified directory." + ) ( "combined-json", po::value()->value_name(boost::join(g_combinedJsonArgs, ",")), - "Output a single json document containing the specified information, can be combined." + "Output a single json document containing the specified information." ) - (g_argAstStr.c_str(), "Outputs the AST of the contract.") - (g_argAstJson.c_str(), "Outputs the AST of the contract in JSON format.") - (g_argAsmStr.c_str(), "Outputs the EVM assembly of the contract.") - (g_argAsmJsonStr.c_str(), "Outputs the EVM assembly of the contract in JSON format.") - (g_argOpcodesStr.c_str(), "Outputs the Opcodes of the contract.") - (g_argBinaryStr.c_str(), "Outputs the contract in binary (hexadecimal).") - (g_argCloneBinaryStr.c_str(), "Output the clone contract in binary (hexadecimal).") - (g_argAbiStr.c_str(), "Outputs the contract's JSON ABI interface.") - (g_argSolInterfaceStr.c_str(), "Outputs the contract's Solidity interface.") - (g_argSignatureHashes.c_str(), "Outputs the contract's functions' signature hashes.") - (g_argGas.c_str(), "Outputs an estimate for each function's maximal gas usage.") - (g_argNatspecUserStr.c_str(), "Outputs the contract's Natspec user documentation.") - (g_argNatspecDevStr.c_str(), "Outputs the contract's Natspec developer documentation."); + (g_argGas.c_str(), "Print an estimate of the maximal gas usage for each function."); + po::options_description outputComponents("Output Components"); + outputComponents.add_options() + (g_argAstStr.c_str(), "AST of all source files.") + (g_argAstJson.c_str(), "AST of all source files in JSON format.") + (g_argAsmStr.c_str(), "EVM assembly of the contracts.") + (g_argAsmJsonStr.c_str(), "EVM assembly of the contracts in JSON format.") + (g_argOpcodesStr.c_str(), "Opcodes of the contracts.") + (g_argBinaryStr.c_str(), "Binary of the contracts in hex.") + (g_argCloneBinaryStr.c_str(), "Binary of the clone contracts in hex.") + (g_argAbiStr.c_str(), "ABI specification of the contracts.") + (g_argSolInterfaceStr.c_str(), "Solidity interface of the contracts.") + (g_argSignatureHashes.c_str(), "Function signature hashes of the contracts.") + (g_argNatspecUserStr.c_str(), "Natspec user documentation of all contracts.") + (g_argNatspecDevStr.c_str(), "Natspec developer documentation of all contracts."); + desc.add(outputComponents); + + po::options_description allOptions = desc; + allOptions.add_options()("input-file", po::value>(), "input file"); // All positional options should be interpreted as input files po::positional_options_description filesPositions; - filesPositions.add("output-dir", 1); filesPositions.add("input-file", -1); // parse the compiler arguments try { - po::store(po::command_line_parser(_argc, _argv).options(desc).positional(filesPositions).allow_unregistered().run(), m_args); - + po::command_line_parser cmdLineParser(_argc, _argv); + cmdLineParser.options(allOptions).positional(filesPositions).allow_unregistered(); + po::store(cmdLineParser.run(), m_args); } catch (po::error const& _exception) { @@ -357,6 +354,18 @@ bool CommandLineInterface::parseArguments(int _argc, char** _argv) return false; } + if (m_args.count("help")) + { + cout << desc; + return false; + } + + if (m_args.count("version")) + { + version(); + return false; + } + if (m_args.count("combined-json")) { vector requests; @@ -369,18 +378,6 @@ bool CommandLineInterface::parseArguments(int _argc, char** _argv) } po::notify(m_args); - if (m_args.count("help")) - { - cout << desc; - return false; - } - - if (m_args.count("version")) - { - version(); - return false; - } - return true; } @@ -414,13 +411,13 @@ bool CommandLineInterface::processInput() m_sourceCodes[infile] = dev::contentsString(infile); } - m_compiler.reset(new CompilerStack(m_args["add-std"].as())); + m_compiler.reset(new CompilerStack(m_args.count(g_argAddStandard) > 0)); try { for (auto const& sourceCode: m_sourceCodes) m_compiler->addSource(sourceCode.first, sourceCode.second); // TODO: Perhaps we should not compile unless requested - bool optimize = m_args["optimize"].as(); + bool optimize = m_args.count("optimize") > 0; unsigned runs = m_args["optimize-runs"].as(); m_compiler->compile(optimize, runs); } @@ -561,7 +558,8 @@ void CommandLineInterface::handleAst(string const& _argStr) converter.print(data); postfix += "_json"; } - createFile(sourceCode.first + postfix + ".ast", data.str()); + boost::filesystem::path path(sourceCode.first); + createFile(path.filename().string() + postfix + ".ast", data.str()); } } else diff --git a/solc/main.cpp b/solc/main.cpp index c5f72980d..eaada1c4e 100644 --- a/solc/main.cpp +++ b/solc/main.cpp @@ -21,6 +21,10 @@ */ #include "CommandLineInterface.h" +#include +#include + +using namespace std; int main(int argc, char** argv) { @@ -29,7 +33,15 @@ int main(int argc, char** argv) return 1; if (!cli.processInput()) return 1; - cli.actOnInput(); + try + { + cli.actOnInput(); + } + catch (boost::exception const& _exception) + { + cerr << "Exception during output generation: " << boost::diagnostic_information(_exception) << endl; + return 1; + } return 0; }