/* 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 . */ // SPDX-License-Identifier: GPL-3.0 /** * @author Alex Beregszaszi * @date 2016 * Standard JSON compiler interface. */ #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include using namespace std; using namespace solidity; using namespace solidity::yul; using namespace solidity::frontend; using namespace solidity::langutil; namespace { Json::Value formatError( Error::Type _type, string const& _component, string const& _message, string const& _formattedMessage = "", Json::Value const& _sourceLocation = Json::Value(), Json::Value const& _secondarySourceLocation = Json::Value() ) { Json::Value error{Json::objectValue}; error["type"] = Error::formatErrorType(_type); error["component"] = _component; error["severity"] = Error::formatErrorSeverityLowercase(Error::errorSeverity(_type)); error["message"] = _message; error["formattedMessage"] = (_formattedMessage.length() > 0) ? _formattedMessage : _message; if (_sourceLocation.isObject()) error["sourceLocation"] = _sourceLocation; if (_secondarySourceLocation.isArray()) error["secondarySourceLocations"] = _secondarySourceLocation; return error; } Json::Value formatFatalError(Error::Type _type, string const& _message) { Json::Value output{Json::objectValue}; output["errors"] = Json::arrayValue; output["errors"].append(formatError(_type, "general", _message)); return output; } Json::Value formatSourceLocation(SourceLocation const* location) { if (!location || !location->sourceName) return Json::nullValue; Json::Value sourceLocation{Json::objectValue}; sourceLocation["file"] = *location->sourceName; sourceLocation["start"] = location->start; sourceLocation["end"] = location->end; return sourceLocation; } Json::Value formatSecondarySourceLocation(SecondarySourceLocation const* _secondaryLocation) { if (!_secondaryLocation) return Json::nullValue; Json::Value secondarySourceLocation{Json::arrayValue}; for (auto const& location: _secondaryLocation->infos) { Json::Value msg = formatSourceLocation(&location.second); msg["message"] = location.first; secondarySourceLocation.append(msg); } return secondarySourceLocation; } Json::Value formatErrorWithException( CharStreamProvider const& _charStreamProvider, util::Exception const& _exception, Error::Type _type, string const& _component, string const& _message, optional _errorId = nullopt ) { string message; // TODO: consider enabling color string formattedMessage = SourceReferenceFormatter::formatExceptionInformation( _exception, _type, _charStreamProvider, false, // colored false, // _withErrorIds true // _printFullType ); if (string const* description = _exception.comment()) message = ((_message.length() > 0) ? (_message + ":") : "") + *description; else message = _message; Json::Value error = formatError( _type, _component, message, formattedMessage, formatSourceLocation(boost::get_error_info(_exception)), formatSecondarySourceLocation(boost::get_error_info(_exception)) ); if (_errorId) error["errorCode"] = to_string(_errorId.value().error); return error; } map> requestedContractNames(Json::Value const& _outputSelection) { map> contracts; for (auto const& sourceName: _outputSelection.getMemberNames()) { string key = (sourceName == "*") ? "" : sourceName; for (auto const& contractName: _outputSelection[sourceName].getMemberNames()) { string value = (contractName == "*") ? "" : contractName; contracts[key].insert(value); } } return contracts; } /// Returns true iff @a _hash (hex with 0x prefix) is the Keccak256 hash of the binary data in @a _content. bool hashMatchesContent(string const& _hash, string const& _content) { try { return util::h256(_hash) == util::keccak256(_content); } catch (util::BadHexCharacter const&) { return false; } } bool isArtifactRequested(Json::Value const& _outputSelection, string const& _artifact, bool _wildcardMatchesExperimental) { static set experimental{"ir", "irOptimized", "wast", "ewasm", "ewasm.wast"}; for (auto const& selectedArtifactJson: _outputSelection) { string const& selectedArtifact = selectedArtifactJson.asString(); if ( _artifact == selectedArtifact || boost::algorithm::starts_with(_artifact, selectedArtifact + ".") ) return true; else if (selectedArtifact == "*") { // "ir", "irOptimized", "wast" and "ewasm.wast" can only be matched by "*" if activated. if (experimental.count(_artifact) == 0 || _wildcardMatchesExperimental) return true; } } return false; } /// /// @a _outputSelection is a JSON object containing a two-level hashmap, where the first level is the filename, /// the second level is the contract name and the value is an array of artifact names to be requested for that contract. /// @a _file is the current file /// @a _contract is the current contract /// @a _artifact is the current artifact name /// /// @returns true if the @a _outputSelection has a match for the requested target in the specific file / contract. /// /// In @a _outputSelection the use of '*' as a wildcard is permitted. /// /// @TODO optimise this. Perhaps flatten the structure upfront. /// bool isArtifactRequested(Json::Value const& _outputSelection, string const& _file, string const& _contract, string const& _artifact, bool _wildcardMatchesExperimental) { if (!_outputSelection.isObject()) return false; for (auto const& file: { _file, string("*") }) if (_outputSelection.isMember(file) && _outputSelection[file].isObject()) { /// For SourceUnit-level targets (such as AST) only allow empty name, otherwise /// for Contract-level targets try both contract name and wildcard vector contracts{ _contract }; if (!_contract.empty()) contracts.emplace_back("*"); for (auto const& contract: contracts) if ( _outputSelection[file].isMember(contract) && _outputSelection[file][contract].isArray() && isArtifactRequested(_outputSelection[file][contract], _artifact, _wildcardMatchesExperimental) ) return true; } return false; } bool isArtifactRequested(Json::Value const& _outputSelection, string const& _file, string const& _contract, vector const& _artifacts, bool _wildcardMatchesExperimental) { for (auto const& artifact: _artifacts) if (isArtifactRequested(_outputSelection, _file, _contract, artifact, _wildcardMatchesExperimental)) return true; return false; } /// @returns all artifact names of the EVM object, either for creation or deploy time. vector evmObjectComponents(string const& _objectKind) { solAssert(_objectKind == "bytecode" || _objectKind == "deployedBytecode", ""); vector components{"", ".object", ".opcodes", ".sourceMap", ".functionDebugData", ".generatedSources", ".linkReferences"}; if (_objectKind == "deployedBytecode") components.push_back(".immutableReferences"); return util::applyMap(components, [&](auto const& _s) { return "evm." + _objectKind + _s; }); } /// @returns true if any binary was requested, i.e. we actually have to perform compilation. bool isBinaryRequested(Json::Value const& _outputSelection) { if (!_outputSelection.isObject()) return false; // This does not include "evm.methodIdentifiers" on purpose! static vector const outputsThatRequireBinaries = vector{ "*", "ir", "irOptimized", "wast", "wasm", "ewasm.wast", "ewasm.wasm", "evm.gasEstimates", "evm.legacyAssembly", "evm.assembly" } + evmObjectComponents("bytecode") + evmObjectComponents("deployedBytecode"); for (auto const& fileRequests: _outputSelection) for (auto const& requests: fileRequests) for (auto const& output: outputsThatRequireBinaries) if (isArtifactRequested(requests, output, false)) return true; return false; } /// @returns true if EVM bytecode was requested, i.e. we have to run the old code generator. bool isEvmBytecodeRequested(Json::Value const& _outputSelection) { if (!_outputSelection.isObject()) return false; static vector const outputsThatRequireEvmBinaries = vector{ "*", "evm.gasEstimates", "evm.legacyAssembly", "evm.assembly" } + evmObjectComponents("bytecode") + evmObjectComponents("deployedBytecode"); for (auto const& fileRequests: _outputSelection) for (auto const& requests: fileRequests) for (auto const& output: outputsThatRequireEvmBinaries) if (isArtifactRequested(requests, output, false)) return true; return false; } /// @returns true if any Ewasm code was requested. Note that as an exception, '*' does not /// yet match "ewasm.wast" or "ewasm" bool isEwasmRequested(Json::Value const& _outputSelection) { if (!_outputSelection.isObject()) return false; for (auto const& fileRequests: _outputSelection) for (auto const& requests: fileRequests) for (auto const& request: requests) if (request == "ewasm" || request == "ewasm.wast") return true; return false; } /// @returns true if any Yul IR was requested. Note that as an exception, '*' does not /// yet match "ir" or "irOptimized" bool isIRRequested(Json::Value const& _outputSelection) { if (isEwasmRequested(_outputSelection)) return true; if (!_outputSelection.isObject()) return false; for (auto const& fileRequests: _outputSelection) for (auto const& requests: fileRequests) for (auto const& request: requests) if (request == "ir" || request == "irOptimized") return true; return false; } Json::Value formatLinkReferences(std::map const& linkReferences) { Json::Value ret{Json::objectValue}; for (auto const& ref: linkReferences) { string const& fullname = ref.second; // If the link reference does not contain a colon, assume that the file name is missing and // the whole string represents the library name. size_t colon = fullname.rfind(':'); string file = (colon != string::npos ? fullname.substr(0, colon) : ""); string name = (colon != string::npos ? fullname.substr(colon + 1) : fullname); Json::Value fileObject = ret.get(file, Json::objectValue); Json::Value libraryArray = fileObject.get(name, Json::arrayValue); Json::Value entry{Json::objectValue}; entry["start"] = Json::UInt(ref.first); entry["length"] = 20; libraryArray.append(entry); fileObject[name] = libraryArray; ret[file] = fileObject; } return ret; } Json::Value formatImmutableReferences(map>> const& _immutableReferences) { Json::Value ret{Json::objectValue}; for (auto const& immutableReference: _immutableReferences) { auto const& [identifier, byteOffsets] = immutableReference.second; Json::Value array(Json::arrayValue); for (size_t byteOffset: byteOffsets) { Json::Value byteRange{Json::objectValue}; byteRange["start"] = Json::UInt(byteOffset); byteRange["length"] = Json::UInt(32); // immutable references are currently always 32 bytes wide array.append(byteRange); } ret[identifier] = array; } return ret; } Json::Value collectEVMObject( evmasm::LinkerObject const& _object, string const* _sourceMap, Json::Value _generatedSources, bool _runtimeObject, function const& _artifactRequested ) { Json::Value output{Json::objectValue}; if (_artifactRequested("object")) output["object"] = _object.toHex(); if (_artifactRequested("opcodes")) output["opcodes"] = evmasm::disassemble(_object.bytecode); if (_artifactRequested("sourceMap")) output["sourceMap"] = _sourceMap ? *_sourceMap : ""; if (_artifactRequested("functionDebugData")) output["functionDebugData"] = StandardCompiler::formatFunctionDebugData(_object.functionDebugData); if (_artifactRequested("linkReferences")) output["linkReferences"] = formatLinkReferences(_object.linkReferences); if (_runtimeObject && _artifactRequested("immutableReferences")) output["immutableReferences"] = formatImmutableReferences(_object.immutableReferences); if (_artifactRequested("generatedSources")) output["generatedSources"] = std::move(_generatedSources); return output; } std::optional checkKeys(Json::Value const& _input, set const& _keys, string const& _name) { if (!!_input && !_input.isObject()) return formatFatalError(Error::Type::JSONError, "\"" + _name + "\" must be an object"); for (auto const& member: _input.getMemberNames()) if (!_keys.count(member)) return formatFatalError(Error::Type::JSONError, "Unknown key \"" + member + "\""); return std::nullopt; } std::optional checkRootKeys(Json::Value const& _input) { static set keys{"auxiliaryInput", "language", "settings", "sources"}; return checkKeys(_input, keys, "root"); } std::optional checkSourceKeys(Json::Value const& _input, string const& _name) { static set keys{"content", "keccak256", "urls"}; return checkKeys(_input, keys, "sources." + _name); } std::optional checkAuxiliaryInputKeys(Json::Value const& _input) { static set keys{"smtlib2responses"}; return checkKeys(_input, keys, "auxiliaryInput"); } std::optional checkSettingsKeys(Json::Value const& _input) { static set keys{"parserErrorRecovery", "debug", "evmVersion", "libraries", "metadata", "modelChecker", "optimizer", "outputSelection", "remappings", "stopAfter", "viaIR"}; return checkKeys(_input, keys, "settings"); } std::optional checkModelCheckerSettingsKeys(Json::Value const& _input) { static set keys{"contracts", "divModNoSlacks", "engine", "invariants", "showUnproved", "solvers", "targets", "timeout"}; return checkKeys(_input, keys, "modelChecker"); } std::optional checkOptimizerKeys(Json::Value const& _input) { static set keys{"details", "enabled", "runs"}; return checkKeys(_input, keys, "settings.optimizer"); } std::optional checkOptimizerDetailsKeys(Json::Value const& _input) { static set keys{"peephole", "inliner", "jumpdestRemover", "orderLiterals", "deduplicate", "cse", "constantOptimizer", "yul", "yulDetails"}; return checkKeys(_input, keys, "settings.optimizer.details"); } std::optional checkOptimizerDetail(Json::Value const& _details, std::string const& _name, bool& _setting) { if (_details.isMember(_name)) { if (!_details[_name].isBool()) return formatFatalError(Error::Type::JSONError, "\"settings.optimizer.details." + _name + "\" must be Boolean"); _setting = _details[_name].asBool(); } return {}; } std::optional checkOptimizerDetailSteps(Json::Value const& _details, std::string const& _name, string& _optimiserSetting, string& _cleanupSetting) { if (_details.isMember(_name)) { if (_details[_name].isString()) { try { yul::OptimiserSuite::validateSequence(_details[_name].asString()); } catch (yul::OptimizerException const& _exception) { return formatFatalError( Error::Type::JSONError, "Invalid optimizer step sequence in \"settings.optimizer.details." + _name + "\": " + _exception.what() ); } string const fullSequence = _details[_name].asString(); auto const delimiterPos = fullSequence.find(":"); _optimiserSetting = fullSequence.substr(0, delimiterPos); if (delimiterPos != string::npos) _cleanupSetting = fullSequence.substr(delimiterPos + 1); else solAssert(_cleanupSetting == OptimiserSettings::DefaultYulOptimiserCleanupSteps); } else return formatFatalError(Error::Type::JSONError, "\"settings.optimizer.details." + _name + "\" must be a string"); } return {}; } std::optional checkMetadataKeys(Json::Value const& _input) { if (_input.isObject()) { if (_input.isMember("useLiteralContent") && !_input["useLiteralContent"].isBool()) return formatFatalError(Error::Type::JSONError, "\"settings.metadata.useLiteralContent\" must be Boolean"); static set hashes{"ipfs", "bzzr1", "none"}; if (_input.isMember("bytecodeHash") && !hashes.count(_input["bytecodeHash"].asString())) return formatFatalError(Error::Type::JSONError, "\"settings.metadata.bytecodeHash\" must be \"ipfs\", \"bzzr1\" or \"none\""); } static set keys{"useLiteralContent", "bytecodeHash"}; return checkKeys(_input, keys, "settings.metadata"); } std::optional checkOutputSelection(Json::Value const& _outputSelection) { if (!!_outputSelection && !_outputSelection.isObject()) return formatFatalError(Error::Type::JSONError, "\"settings.outputSelection\" must be an object"); for (auto const& sourceName: _outputSelection.getMemberNames()) { auto const& sourceVal = _outputSelection[sourceName]; if (!sourceVal.isObject()) return formatFatalError( Error::Type::JSONError, "\"settings.outputSelection." + sourceName + "\" must be an object" ); for (auto const& contractName: sourceVal.getMemberNames()) { auto const& contractVal = sourceVal[contractName]; if (!contractVal.isArray()) return formatFatalError( Error::Type::JSONError, "\"settings.outputSelection." + sourceName + "." + contractName + "\" must be a string array" ); for (auto const& output: contractVal) if (!output.isString()) return formatFatalError( Error::Type::JSONError, "\"settings.outputSelection." + sourceName + "." + contractName + "\" must be a string array" ); } } return std::nullopt; } /// Validates the optimizer settings and returns them in a parsed object. /// On error returns the json-formatted error message. std::variant parseOptimizerSettings(Json::Value const& _jsonInput) { if (auto result = checkOptimizerKeys(_jsonInput)) return *result; OptimiserSettings settings = OptimiserSettings::minimal(); if (_jsonInput.isMember("enabled")) { if (!_jsonInput["enabled"].isBool()) return formatFatalError(Error::Type::JSONError, "The \"enabled\" setting must be a Boolean."); if (_jsonInput["enabled"].asBool()) settings = OptimiserSettings::standard(); } if (_jsonInput.isMember("runs")) { if (!_jsonInput["runs"].isUInt()) return formatFatalError(Error::Type::JSONError, "The \"runs\" setting must be an unsigned number."); settings.expectedExecutionsPerDeployment = _jsonInput["runs"].asUInt(); } if (_jsonInput.isMember("details")) { Json::Value const& details = _jsonInput["details"]; if (auto result = checkOptimizerDetailsKeys(details)) return *result; if (auto error = checkOptimizerDetail(details, "peephole", settings.runPeephole)) return *error; if (auto error = checkOptimizerDetail(details, "inliner", settings.runInliner)) return *error; if (auto error = checkOptimizerDetail(details, "jumpdestRemover", settings.runJumpdestRemover)) return *error; if (auto error = checkOptimizerDetail(details, "orderLiterals", settings.runOrderLiterals)) return *error; if (auto error = checkOptimizerDetail(details, "deduplicate", settings.runDeduplicate)) return *error; if (auto error = checkOptimizerDetail(details, "cse", settings.runCSE)) return *error; if (auto error = checkOptimizerDetail(details, "constantOptimizer", settings.runConstantOptimiser)) return *error; if (auto error = checkOptimizerDetail(details, "yul", settings.runYulOptimiser)) return *error; settings.optimizeStackAllocation = settings.runYulOptimiser; if (details.isMember("yulDetails")) { if (!settings.runYulOptimiser) return formatFatalError(Error::Type::JSONError, "\"Providing yulDetails requires Yul optimizer to be enabled."); if (auto result = checkKeys(details["yulDetails"], {"stackAllocation", "optimizerSteps"}, "settings.optimizer.details.yulDetails")) return *result; if (auto error = checkOptimizerDetail(details["yulDetails"], "stackAllocation", settings.optimizeStackAllocation)) return *error; if (auto error = checkOptimizerDetailSteps(details["yulDetails"], "optimizerSteps", settings.yulOptimiserSteps, settings.yulOptimiserCleanupSteps)) return *error; } } return { std::move(settings) }; } } std::variant StandardCompiler::parseInput(Json::Value const& _input) { InputsAndSettings ret; if (!_input.isObject()) return formatFatalError(Error::Type::JSONError, "Input is not a JSON object."); if (auto result = checkRootKeys(_input)) return *result; ret.language = _input["language"].asString(); Json::Value const& sources = _input["sources"]; if (!sources.isObject() && !sources.isNull()) return formatFatalError(Error::Type::JSONError, "\"sources\" is not a JSON object."); if (sources.empty()) return formatFatalError(Error::Type::JSONError, "No input sources specified."); ret.errors = Json::arrayValue; for (auto const& sourceName: sources.getMemberNames()) { string hash; if (auto result = checkSourceKeys(sources[sourceName], sourceName)) return *result; if (sources[sourceName]["keccak256"].isString()) hash = sources[sourceName]["keccak256"].asString(); if (sources[sourceName]["content"].isString()) { string content = sources[sourceName]["content"].asString(); if (!hash.empty() && !hashMatchesContent(hash, content)) ret.errors.append(formatError( Error::Type::IOError, "general", "Mismatch between content and supplied hash for \"" + sourceName + "\"" )); else ret.sources[sourceName] = content; } else if (sources[sourceName]["urls"].isArray()) { if (!m_readFile) return formatFatalError(Error::Type::JSONError, "No import callback supplied, but URL is requested."); vector failures; bool found = false; for (auto const& url: sources[sourceName]["urls"]) { if (!url.isString()) return formatFatalError(Error::Type::JSONError, "URL must be a string."); ReadCallback::Result result = m_readFile(ReadCallback::kindString(ReadCallback::Kind::ReadFile), url.asString()); if (result.success) { if (!hash.empty() && !hashMatchesContent(hash, result.responseOrErrorMessage)) ret.errors.append(formatError( Error::Type::IOError, "general", "Mismatch between content and supplied hash for \"" + sourceName + "\" at \"" + url.asString() + "\"" )); else { ret.sources[sourceName] = result.responseOrErrorMessage; found = true; break; } } else failures.push_back("Cannot import url (\"" + url.asString() + "\"): " + result.responseOrErrorMessage); } for (auto const& failure: failures) { /// If the import succeeded, let mark all the others as warnings, otherwise all of them are errors. ret.errors.append(formatError( found ? Error::Type::Warning : Error::Type::IOError, "general", failure )); } } else return formatFatalError(Error::Type::JSONError, "Invalid input source specified."); } Json::Value const& auxInputs = _input["auxiliaryInput"]; if (auto result = checkAuxiliaryInputKeys(auxInputs)) return *result; if (!!auxInputs) { Json::Value const& smtlib2Responses = auxInputs["smtlib2responses"]; if (!!smtlib2Responses) { if (!smtlib2Responses.isObject()) return formatFatalError(Error::Type::JSONError, "\"auxiliaryInput.smtlib2responses\" must be an object."); for (auto const& hashString: smtlib2Responses.getMemberNames()) { util::h256 hash; try { hash = util::h256(hashString); } catch (util::BadHexCharacter const&) { return formatFatalError(Error::Type::JSONError, "Invalid hex encoding of SMTLib2 auxiliary input."); } if (!smtlib2Responses[hashString].isString()) return formatFatalError( Error::Type::JSONError, "\"smtlib2Responses." + hashString + "\" must be a string." ); ret.smtLib2Responses[hash] = smtlib2Responses[hashString].asString(); } } } Json::Value const& settings = _input.get("settings", Json::Value()); if (auto result = checkSettingsKeys(settings)) return *result; if (settings.isMember("stopAfter")) { if (!settings["stopAfter"].isString()) return formatFatalError(Error::Type::JSONError, "\"settings.stopAfter\" must be a string."); if (settings["stopAfter"].asString() != "parsing") return formatFatalError(Error::Type::JSONError, "Invalid value for \"settings.stopAfter\". Only valid value is \"parsing\"."); ret.stopAfter = CompilerStack::State::Parsed; } if (settings.isMember("parserErrorRecovery")) { if (!settings["parserErrorRecovery"].isBool()) return formatFatalError(Error::Type::JSONError, "\"settings.parserErrorRecovery\" must be a Boolean."); ret.parserErrorRecovery = settings["parserErrorRecovery"].asBool(); } if (settings.isMember("viaIR")) { if (!settings["viaIR"].isBool()) return formatFatalError(Error::Type::JSONError, "\"settings.viaIR\" must be a Boolean."); ret.viaIR = settings["viaIR"].asBool(); } if (settings.isMember("evmVersion")) { if (!settings["evmVersion"].isString()) return formatFatalError(Error::Type::JSONError, "evmVersion must be a string."); std::optional version = langutil::EVMVersion::fromString(settings["evmVersion"].asString()); if (!version) return formatFatalError(Error::Type::JSONError, "Invalid EVM version requested."); ret.evmVersion = *version; } if (settings.isMember("debug")) { if (auto result = checkKeys(settings["debug"], {"revertStrings", "debugInfo"}, "settings.debug")) return *result; if (settings["debug"].isMember("revertStrings")) { if (!settings["debug"]["revertStrings"].isString()) return formatFatalError(Error::Type::JSONError, "settings.debug.revertStrings must be a string."); std::optional revertStrings = revertStringsFromString(settings["debug"]["revertStrings"].asString()); if (!revertStrings) return formatFatalError(Error::Type::JSONError, "Invalid value for settings.debug.revertStrings."); if (*revertStrings == RevertStrings::VerboseDebug) return formatFatalError( Error::Type::UnimplementedFeatureError, "Only \"default\", \"strip\" and \"debug\" are implemented for settings.debug.revertStrings for now." ); ret.revertStrings = *revertStrings; } if (settings["debug"].isMember("debugInfo")) { if (!settings["debug"]["debugInfo"].isArray()) return formatFatalError(Error::Type::JSONError, "settings.debug.debugInfo must be an array."); vector components; for (Json::Value const& arrayValue: settings["debug"]["debugInfo"]) components.push_back(arrayValue.asString()); optional debugInfoSelection = DebugInfoSelection::fromComponents( components, true /* _acceptWildcards */ ); if (!debugInfoSelection.has_value()) return formatFatalError(Error::Type::JSONError, "Invalid value in settings.debug.debugInfo."); if (debugInfoSelection->snippet && !debugInfoSelection->location) return formatFatalError( Error::Type::JSONError, "To use 'snippet' with settings.debug.debugInfo you must select also 'location'." ); ret.debugInfoSelection = debugInfoSelection.value(); } } if (settings.isMember("remappings") && !settings["remappings"].isArray()) return formatFatalError(Error::Type::JSONError, "\"settings.remappings\" must be an array of strings."); for (auto const& remapping: settings.get("remappings", Json::Value())) { if (!remapping.isString()) return formatFatalError(Error::Type::JSONError, "\"settings.remappings\" must be an array of strings"); if (auto r = ImportRemapper::parseRemapping(remapping.asString())) ret.remappings.emplace_back(std::move(*r)); else return formatFatalError(Error::Type::JSONError, "Invalid remapping: \"" + remapping.asString() + "\""); } if (settings.isMember("optimizer")) { auto optimiserSettings = parseOptimizerSettings(settings["optimizer"]); if (std::holds_alternative(optimiserSettings)) return std::get(std::move(optimiserSettings)); // was an error else ret.optimiserSettings = std::get(std::move(optimiserSettings)); } Json::Value jsonLibraries = settings.get("libraries", Json::Value(Json::objectValue)); if (!jsonLibraries.isObject()) return formatFatalError(Error::Type::JSONError, "\"libraries\" is not a JSON object."); for (auto const& sourceName: jsonLibraries.getMemberNames()) { auto const& jsonSourceName = jsonLibraries[sourceName]; if (!jsonSourceName.isObject()) return formatFatalError(Error::Type::JSONError, "Library entry is not a JSON object."); for (auto const& library: jsonSourceName.getMemberNames()) { if (!jsonSourceName[library].isString()) return formatFatalError(Error::Type::JSONError, "Library address must be a string."); string address = jsonSourceName[library].asString(); if (!boost::starts_with(address, "0x")) return formatFatalError( Error::Type::JSONError, "Library address is not prefixed with \"0x\"." ); if (address.length() != 42) return formatFatalError( Error::Type::JSONError, "Library address is of invalid length." ); try { ret.libraries[sourceName + ":" + library] = util::h160(address); } catch (util::BadHexCharacter const&) { return formatFatalError( Error::Type::JSONError, "Invalid library address (\"" + address + "\") supplied." ); } } } Json::Value metadataSettings = settings.get("metadata", Json::Value()); if (auto result = checkMetadataKeys(metadataSettings)) return *result; ret.metadataLiteralSources = metadataSettings.get("useLiteralContent", Json::Value(false)).asBool(); if (metadataSettings.isMember("bytecodeHash")) { auto metadataHash = metadataSettings["bytecodeHash"].asString(); ret.metadataHash = metadataHash == "ipfs" ? CompilerStack::MetadataHash::IPFS : metadataHash == "bzzr1" ? CompilerStack::MetadataHash::Bzzr1 : CompilerStack::MetadataHash::None; } Json::Value outputSelection = settings.get("outputSelection", Json::Value()); if (auto jsonError = checkOutputSelection(outputSelection)) return *jsonError; ret.outputSelection = std::move(outputSelection); if (ret.stopAfter != CompilerStack::State::CompilationSuccessful && isBinaryRequested(ret.outputSelection)) return formatFatalError( Error::Type::JSONError, "Requested output selection conflicts with \"settings.stopAfter\"." ); Json::Value const& modelCheckerSettings = settings.get("modelChecker", Json::Value()); if (auto result = checkModelCheckerSettingsKeys(modelCheckerSettings)) return *result; if (modelCheckerSettings.isMember("contracts")) { auto const& sources = modelCheckerSettings["contracts"]; if (!sources.isObject() && !sources.isNull()) return formatFatalError(Error::Type::JSONError, "settings.modelChecker.contracts is not a JSON object."); map> sourceContracts; for (auto const& source: sources.getMemberNames()) { if (source.empty()) return formatFatalError(Error::Type::JSONError, "Source name cannot be empty."); auto const& contracts = sources[source]; if (!contracts.isArray()) return formatFatalError(Error::Type::JSONError, "Source contracts must be an array."); for (auto const& contract: contracts) { if (!contract.isString()) return formatFatalError(Error::Type::JSONError, "Every contract in settings.modelChecker.contracts must be a string."); if (contract.asString().empty()) return formatFatalError(Error::Type::JSONError, "Contract name cannot be empty."); sourceContracts[source].insert(contract.asString()); } if (sourceContracts[source].empty()) return formatFatalError(Error::Type::JSONError, "Source contracts must be a non-empty array."); } ret.modelCheckerSettings.contracts = {std::move(sourceContracts)}; } if (modelCheckerSettings.isMember("divModNoSlacks")) { auto const& divModNoSlacks = modelCheckerSettings["divModNoSlacks"]; if (!divModNoSlacks.isBool()) return formatFatalError(Error::Type::JSONError, "settings.modelChecker.divModNoSlacks must be a Boolean."); ret.modelCheckerSettings.divModNoSlacks = divModNoSlacks.asBool(); } if (modelCheckerSettings.isMember("engine")) { if (!modelCheckerSettings["engine"].isString()) return formatFatalError(Error::Type::JSONError, "settings.modelChecker.engine must be a string."); std::optional engine = ModelCheckerEngine::fromString(modelCheckerSettings["engine"].asString()); if (!engine) return formatFatalError(Error::Type::JSONError, "Invalid model checker engine requested."); ret.modelCheckerSettings.engine = *engine; } if (modelCheckerSettings.isMember("invariants")) { auto const& invariantsArray = modelCheckerSettings["invariants"]; if (!invariantsArray.isArray()) return formatFatalError(Error::Type::JSONError, "settings.modelChecker.invariants must be an array."); ModelCheckerInvariants invariants; for (auto const& i: invariantsArray) { if (!i.isString()) return formatFatalError(Error::Type::JSONError, "Every invariant type in settings.modelChecker.invariants must be a string."); if (!invariants.setFromString(i.asString())) return formatFatalError(Error::Type::JSONError, "Invalid model checker invariants requested."); } if (invariants.invariants.empty()) return formatFatalError(Error::Type::JSONError, "settings.modelChecker.invariants must be a non-empty array."); ret.modelCheckerSettings.invariants = invariants; } if (modelCheckerSettings.isMember("showUnproved")) { auto const& showUnproved = modelCheckerSettings["showUnproved"]; if (!showUnproved.isBool()) return formatFatalError(Error::Type::JSONError, "settings.modelChecker.showUnproved must be a Boolean value."); ret.modelCheckerSettings.showUnproved = showUnproved.asBool(); } if (modelCheckerSettings.isMember("solvers")) { auto const& solversArray = modelCheckerSettings["solvers"]; if (!solversArray.isArray()) return formatFatalError(Error::Type::JSONError, "settings.modelChecker.solvers must be an array."); smtutil::SMTSolverChoice solvers; for (auto const& s: solversArray) { if (!s.isString()) return formatFatalError(Error::Type::JSONError, "Every target in settings.modelChecker.solvers must be a string."); if (!solvers.setSolver(s.asString())) return formatFatalError(Error::Type::JSONError, "Invalid model checker solvers requested."); } ret.modelCheckerSettings.solvers = solvers; } if (modelCheckerSettings.isMember("targets")) { auto const& targetsArray = modelCheckerSettings["targets"]; if (!targetsArray.isArray()) return formatFatalError(Error::Type::JSONError, "settings.modelChecker.targets must be an array."); ModelCheckerTargets targets; for (auto const& t: targetsArray) { if (!t.isString()) return formatFatalError(Error::Type::JSONError, "Every target in settings.modelChecker.targets must be a string."); if (!targets.setFromString(t.asString())) return formatFatalError(Error::Type::JSONError, "Invalid model checker targets requested."); } if (targets.targets.empty()) return formatFatalError(Error::Type::JSONError, "settings.modelChecker.targets must be a non-empty array."); ret.modelCheckerSettings.targets = targets; } if (modelCheckerSettings.isMember("timeout")) { if (!modelCheckerSettings["timeout"].isUInt()) return formatFatalError(Error::Type::JSONError, "settings.modelChecker.timeout must be an unsigned integer."); ret.modelCheckerSettings.timeout = modelCheckerSettings["timeout"].asUInt(); } return { std::move(ret) }; } Json::Value StandardCompiler::compileSolidity(StandardCompiler::InputsAndSettings _inputsAndSettings) { CompilerStack compilerStack(m_readFile); StringMap sourceList = std::move(_inputsAndSettings.sources); compilerStack.setSources(sourceList); for (auto const& smtLib2Response: _inputsAndSettings.smtLib2Responses) compilerStack.addSMTLib2Response(smtLib2Response.first, smtLib2Response.second); compilerStack.setViaIR(_inputsAndSettings.viaIR); compilerStack.setEVMVersion(_inputsAndSettings.evmVersion); compilerStack.setParserErrorRecovery(_inputsAndSettings.parserErrorRecovery); compilerStack.setRemappings(std::move(_inputsAndSettings.remappings)); compilerStack.setOptimiserSettings(std::move(_inputsAndSettings.optimiserSettings)); compilerStack.setRevertStringBehaviour(_inputsAndSettings.revertStrings); if (_inputsAndSettings.debugInfoSelection.has_value()) compilerStack.selectDebugInfo(_inputsAndSettings.debugInfoSelection.value()); compilerStack.setLibraries(_inputsAndSettings.libraries); compilerStack.useMetadataLiteralSources(_inputsAndSettings.metadataLiteralSources); compilerStack.setMetadataHash(_inputsAndSettings.metadataHash); compilerStack.setRequestedContractNames(requestedContractNames(_inputsAndSettings.outputSelection)); compilerStack.setModelCheckerSettings(_inputsAndSettings.modelCheckerSettings); compilerStack.enableEvmBytecodeGeneration(isEvmBytecodeRequested(_inputsAndSettings.outputSelection)); compilerStack.enableIRGeneration(isIRRequested(_inputsAndSettings.outputSelection)); compilerStack.enableEwasmGeneration(isEwasmRequested(_inputsAndSettings.outputSelection)); Json::Value errors = std::move(_inputsAndSettings.errors); bool const binariesRequested = isBinaryRequested(_inputsAndSettings.outputSelection); try { if (binariesRequested) compilerStack.compile(); else compilerStack.parseAndAnalyze(_inputsAndSettings.stopAfter); for (auto const& error: compilerStack.errors()) { Error const& err = dynamic_cast(*error); errors.append(formatErrorWithException( compilerStack, *error, err.type(), "general", "", err.errorId() )); } } /// This is only thrown in a very few locations. catch (Error const& _error) { errors.append(formatErrorWithException( compilerStack, _error, _error.type(), "general", "Uncaught error: " )); } /// This should not be leaked from compile(). catch (FatalError const& _exception) { errors.append(formatError( Error::Type::FatalError, "general", "Uncaught fatal error: " + boost::diagnostic_information(_exception) )); } catch (CompilerError const& _exception) { errors.append(formatErrorWithException( compilerStack, _exception, Error::Type::CompilerError, "general", "Compiler error (" + _exception.lineInfo() + ")" )); } catch (InternalCompilerError const& _exception) { errors.append(formatErrorWithException( compilerStack, _exception, Error::Type::InternalCompilerError, "general", "Internal compiler error (" + _exception.lineInfo() + ")" )); } catch (UnimplementedFeatureError const& _exception) { errors.append(formatErrorWithException( compilerStack, _exception, Error::Type::UnimplementedFeatureError, "general", "Unimplemented feature (" + _exception.lineInfo() + ")" )); } catch (yul::YulException const& _exception) { errors.append(formatErrorWithException( compilerStack, _exception, Error::Type::YulException, "general", "Yul exception" )); } catch (smtutil::SMTLogicError const& _exception) { errors.append(formatErrorWithException( compilerStack, _exception, Error::Type::SMTLogicException, "general", "SMT logic exception" )); } catch (util::Exception const& _exception) { errors.append(formatError( Error::Type::Exception, "general", "Exception during compilation: " + boost::diagnostic_information(_exception) )); } catch (std::exception const& _exception) { errors.append(formatError( Error::Type::Exception, "general", "Unknown exception during compilation: " + boost::diagnostic_information(_exception) )); } catch (...) { errors.append(formatError( Error::Type::Exception, "general", "Unknown exception during compilation: " + boost::current_exception_diagnostic_information() )); } bool analysisPerformed = compilerStack.state() >= CompilerStack::State::AnalysisPerformed; bool const compilationSuccess = compilerStack.state() == CompilerStack::State::CompilationSuccessful; if (compilerStack.hasError() && !_inputsAndSettings.parserErrorRecovery) analysisPerformed = false; /// Inconsistent state - stop here to receive error reports from users if ( ((binariesRequested && !compilationSuccess) || !analysisPerformed) && (errors.empty() && _inputsAndSettings.stopAfter >= CompilerStack::State::AnalysisPerformed) ) return formatFatalError(Error::Type::InternalCompilerError, "No error reported, but compilation failed."); Json::Value output = Json::objectValue; if (errors.size() > 0) output["errors"] = std::move(errors); if (!compilerStack.unhandledSMTLib2Queries().empty()) for (string const& query: compilerStack.unhandledSMTLib2Queries()) output["auxiliaryInputRequested"]["smtlib2queries"]["0x" + util::keccak256(query).hex()] = query; bool const wildcardMatchesExperimental = false; output["sources"] = Json::objectValue; unsigned sourceIndex = 0; if (compilerStack.state() >= CompilerStack::State::Parsed && (!compilerStack.hasError() || _inputsAndSettings.parserErrorRecovery)) for (string const& sourceName: compilerStack.sourceNames()) { Json::Value sourceResult = Json::objectValue; sourceResult["id"] = sourceIndex++; if (isArtifactRequested(_inputsAndSettings.outputSelection, sourceName, "", "ast", wildcardMatchesExperimental)) sourceResult["ast"] = ASTJsonExporter(compilerStack.state(), compilerStack.sourceIndices()).toJson(compilerStack.ast(sourceName)); output["sources"][sourceName] = sourceResult; } Json::Value contractsOutput = Json::objectValue; for (string const& contractName: analysisPerformed ? compilerStack.contractNames() : vector()) { size_t colon = contractName.rfind(':'); solAssert(colon != string::npos, ""); string file = contractName.substr(0, colon); string name = contractName.substr(colon + 1); // ABI, storage layout, documentation and metadata Json::Value contractData(Json::objectValue); if (isArtifactRequested(_inputsAndSettings.outputSelection, file, name, "abi", wildcardMatchesExperimental)) contractData["abi"] = compilerStack.contractABI(contractName); if (isArtifactRequested(_inputsAndSettings.outputSelection, file, name, "storageLayout", false)) contractData["storageLayout"] = compilerStack.storageLayout(contractName); if (isArtifactRequested(_inputsAndSettings.outputSelection, file, name, "metadata", wildcardMatchesExperimental)) contractData["metadata"] = compilerStack.metadata(contractName); if (isArtifactRequested(_inputsAndSettings.outputSelection, file, name, "userdoc", wildcardMatchesExperimental)) contractData["userdoc"] = compilerStack.natspecUser(contractName); if (isArtifactRequested(_inputsAndSettings.outputSelection, file, name, "devdoc", wildcardMatchesExperimental)) contractData["devdoc"] = compilerStack.natspecDev(contractName); // IR if (compilationSuccess && isArtifactRequested(_inputsAndSettings.outputSelection, file, name, "ir", wildcardMatchesExperimental)) contractData["ir"] = compilerStack.yulIR(contractName); if (compilationSuccess && isArtifactRequested(_inputsAndSettings.outputSelection, file, name, "irOptimized", wildcardMatchesExperimental)) contractData["irOptimized"] = compilerStack.yulIROptimized(contractName); // Ewasm if (compilationSuccess && isArtifactRequested(_inputsAndSettings.outputSelection, file, name, "ewasm.wast", wildcardMatchesExperimental)) contractData["ewasm"]["wast"] = compilerStack.ewasm(contractName); if (compilationSuccess && isArtifactRequested(_inputsAndSettings.outputSelection, file, name, "ewasm.wasm", wildcardMatchesExperimental)) contractData["ewasm"]["wasm"] = compilerStack.ewasmObject(contractName).toHex(); // EVM Json::Value evmData(Json::objectValue); if (compilationSuccess && isArtifactRequested(_inputsAndSettings.outputSelection, file, name, "evm.assembly", wildcardMatchesExperimental)) evmData["assembly"] = compilerStack.assemblyString(contractName, sourceList); if (compilationSuccess && isArtifactRequested(_inputsAndSettings.outputSelection, file, name, "evm.legacyAssembly", wildcardMatchesExperimental)) evmData["legacyAssembly"] = compilerStack.assemblyJSON(contractName); if (isArtifactRequested(_inputsAndSettings.outputSelection, file, name, "evm.methodIdentifiers", wildcardMatchesExperimental)) evmData["methodIdentifiers"] = compilerStack.interfaceSymbols(contractName)["methods"]; if (compilationSuccess && isArtifactRequested(_inputsAndSettings.outputSelection, file, name, "evm.gasEstimates", wildcardMatchesExperimental)) evmData["gasEstimates"] = compilerStack.gasEstimates(contractName); if (compilationSuccess && isArtifactRequested( _inputsAndSettings.outputSelection, file, name, evmObjectComponents("bytecode"), wildcardMatchesExperimental )) evmData["bytecode"] = collectEVMObject( compilerStack.object(contractName), compilerStack.sourceMapping(contractName), compilerStack.generatedSources(contractName), false, [&](string const& _element) { return isArtifactRequested( _inputsAndSettings.outputSelection, file, name, "evm.bytecode." + _element, wildcardMatchesExperimental ); } ); if (compilationSuccess && isArtifactRequested( _inputsAndSettings.outputSelection, file, name, evmObjectComponents("deployedBytecode"), wildcardMatchesExperimental )) evmData["deployedBytecode"] = collectEVMObject( compilerStack.runtimeObject(contractName), compilerStack.runtimeSourceMapping(contractName), compilerStack.generatedSources(contractName, true), true, [&](string const& _element) { return isArtifactRequested( _inputsAndSettings.outputSelection, file, name, "evm.deployedBytecode." + _element, wildcardMatchesExperimental ); } ); if (!evmData.empty()) contractData["evm"] = evmData; if (!contractData.empty()) { if (!contractsOutput.isMember(file)) contractsOutput[file] = Json::objectValue; contractsOutput[file][name] = contractData; } } if (!contractsOutput.empty()) output["contracts"] = contractsOutput; return output; } Json::Value StandardCompiler::compileYul(InputsAndSettings _inputsAndSettings) { Json::Value output = Json::objectValue; output["errors"] = std::move(_inputsAndSettings.errors); if (_inputsAndSettings.sources.size() != 1) { output["errors"].append(formatError( Error::Type::JSONError, "general", "Yul mode only supports exactly one input file." )); return output; } if (!_inputsAndSettings.smtLib2Responses.empty()) { output["errors"].append(formatError( Error::Type::JSONError, "general", "Yul mode does not support smtlib2responses." )); return output; } if (!_inputsAndSettings.remappings.empty()) { output["errors"].append(formatError( Error::Type::JSONError, "general", "Field \"settings.remappings\" cannot be used for Yul." )); return output; } if (_inputsAndSettings.revertStrings != RevertStrings::Default) { output["errors"].append(formatError( Error::Type::JSONError, "general", "Field \"settings.debug.revertStrings\" cannot be used for Yul." )); return output; } YulStack stack( _inputsAndSettings.evmVersion, YulStack::Language::StrictAssembly, _inputsAndSettings.optimiserSettings, _inputsAndSettings.debugInfoSelection.has_value() ? _inputsAndSettings.debugInfoSelection.value() : DebugInfoSelection::Default() ); string const& sourceName = _inputsAndSettings.sources.begin()->first; string const& sourceContents = _inputsAndSettings.sources.begin()->second; // Inconsistent state - stop here to receive error reports from users if (!stack.parseAndAnalyze(sourceName, sourceContents) && stack.errors().empty()) { output["errors"].append(formatError( Error::Type::InternalCompilerError, "general", "No error reported, but compilation failed." )); return output; } if (!stack.errors().empty()) { for (auto const& error: stack.errors()) { auto err = dynamic_pointer_cast(error); output["errors"].append(formatErrorWithException( stack, *error, err->type(), "general", "" )); } return output; } string contractName = stack.parserResult()->name.str(); bool const wildcardMatchesExperimental = true; if (isArtifactRequested(_inputsAndSettings.outputSelection, sourceName, contractName, "ir", wildcardMatchesExperimental)) output["contracts"][sourceName][contractName]["ir"] = stack.print(); stack.optimize(); MachineAssemblyObject object; MachineAssemblyObject deployedObject; tie(object, deployedObject) = stack.assembleWithDeployed(); if (object.bytecode) object.bytecode->link(_inputsAndSettings.libraries); if (deployedObject.bytecode) deployedObject.bytecode->link(_inputsAndSettings.libraries); for (auto&& [kind, isDeployed]: {make_pair("bytecode"s, false), make_pair("deployedBytecode"s, true)}) if (isArtifactRequested( _inputsAndSettings.outputSelection, sourceName, contractName, evmObjectComponents(kind), wildcardMatchesExperimental )) { MachineAssemblyObject const& o = isDeployed ? deployedObject : object; if (o.bytecode) output["contracts"][sourceName][contractName]["evm"][kind] = collectEVMObject( *o.bytecode, o.sourceMappings.get(), Json::arrayValue, isDeployed, [&, kind = kind](string const& _element) { return isArtifactRequested( _inputsAndSettings.outputSelection, sourceName, contractName, "evm." + kind + "." + _element, wildcardMatchesExperimental ); } ); } if (isArtifactRequested(_inputsAndSettings.outputSelection, sourceName, contractName, "irOptimized", wildcardMatchesExperimental)) output["contracts"][sourceName][contractName]["irOptimized"] = stack.print(); if (isArtifactRequested(_inputsAndSettings.outputSelection, sourceName, contractName, "evm.assembly", wildcardMatchesExperimental)) output["contracts"][sourceName][contractName]["evm"]["assembly"] = object.assembly; return output; } Json::Value StandardCompiler::compile(Json::Value const& _input) noexcept { YulStringRepository::reset(); try { auto parsed = parseInput(_input); if (std::holds_alternative(parsed)) return std::get(std::move(parsed)); InputsAndSettings settings = std::get(std::move(parsed)); if (settings.language == "Solidity") return compileSolidity(std::move(settings)); else if (settings.language == "Yul") return compileYul(std::move(settings)); else return formatFatalError(Error::Type::JSONError, "Only \"Solidity\" or \"Yul\" is supported as a language."); } catch (Json::LogicError const& _exception) { return formatFatalError(Error::Type::InternalCompilerError, string("JSON logic exception: ") + _exception.what()); } catch (Json::RuntimeError const& _exception) { return formatFatalError(Error::Type::InternalCompilerError, string("JSON runtime exception: ") + _exception.what()); } catch (util::Exception const& _exception) { return formatFatalError(Error::Type::InternalCompilerError, "Internal exception in StandardCompiler::compile: " + boost::diagnostic_information(_exception)); } catch (...) { return formatFatalError(Error::Type::InternalCompilerError, "Internal exception in StandardCompiler::compile: " + boost::current_exception_diagnostic_information()); } } string StandardCompiler::compile(string const& _input) noexcept { Json::Value input; string errors; try { if (!util::jsonParseStrict(_input, input, &errors)) return util::jsonPrint(formatFatalError(Error::Type::JSONError, errors), m_jsonPrintingFormat); } catch (...) { return "{\"errors\":[{\"type\":\"JSONError\",\"component\":\"general\",\"severity\":\"error\",\"message\":\"Error parsing input JSON.\"}]}"; } // cout << "Input: " << input.toStyledString() << endl; Json::Value output = compile(input); // cout << "Output: " << output.toStyledString() << endl; try { return util::jsonPrint(output, m_jsonPrintingFormat); } catch (...) { return "{\"errors\":[{\"type\":\"JSONError\",\"component\":\"general\",\"severity\":\"error\",\"message\":\"Error writing output JSON.\"}]}"; } } Json::Value StandardCompiler::formatFunctionDebugData( map const& _debugInfo ) { Json::Value ret(Json::objectValue); for (auto const& [name, info]: _debugInfo) { Json::Value fun; if (info.sourceID) fun["id"] = Json::UInt64(*info.sourceID); else fun["id"] = Json::nullValue; if (info.bytecodeOffset) fun["entryPoint"] = Json::UInt64(*info.bytecodeOffset); else fun["entryPoint"] = Json::nullValue; fun["parameterSlots"] = Json::UInt64(info.params); fun["returnSlots"] = Json::UInt64(info.returns); ret[name] = std::move(fun); } return ret; }