/* 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 /** * Yul code and data object container. */ #include #include #include #include #include #include #include #include #include #include #include #include using namespace solidity; using namespace solidity::langutil; using namespace solidity::util; using namespace solidity::yul; namespace { std::string indent(std::string const& _input) { if (_input.empty()) return _input; return boost::replace_all_copy(" " + _input, "\n", "\n "); } } std::string Data::toString(Dialect const*, DebugInfoSelection const&, CharStreamProvider const*) const { return "data \"" + name.str() + "\" hex\"" + util::toHex(data) + "\""; } std::string Object::toString( Dialect const* _dialect, DebugInfoSelection const& _debugInfoSelection, CharStreamProvider const* _soliditySourceProvider ) const { yulAssert(code, "No code"); yulAssert(debugData, "No debug data"); std::string useSrcComment; if (debugData->sourceNames) useSrcComment = "/// @use-src " + joinHumanReadable(ranges::views::transform(*debugData->sourceNames, [](auto&& _pair) { return std::to_string(_pair.first) + ":" + util::escapeAndQuoteString(*_pair.second); })) + "\n"; std::string inner = "code " + AsmPrinter( _dialect, debugData->sourceNames, _debugInfoSelection, _soliditySourceProvider )(*code); for (auto const& obj: subObjects) inner += "\n" + obj->toString(_dialect, _debugInfoSelection, _soliditySourceProvider); return useSrcComment + "object \"" + name.str() + "\" {\n" + indent(inner) + "\n}"; } Json::Value Data::toJson() const { Json::Value ret{Json::objectValue}; ret["nodeType"] = "YulData"; ret["value"] = util::toHex(data); return ret; } Json::Value Object::toJson() const { yulAssert(code, "No code"); Json::Value codeJson{Json::objectValue}; codeJson["nodeType"] = "YulCode"; codeJson["block"] = AsmJsonConverter(0 /* sourceIndex */)(*code); Json::Value subObjectsJson{Json::arrayValue}; for (std::shared_ptr const& subObject: subObjects) subObjectsJson.append(subObject->toJson()); Json::Value ret{Json::objectValue}; ret["nodeType"] = "YulObject"; ret["name"] = name.str(); ret["code"] = codeJson; ret["subObjects"] = subObjectsJson; return ret; } std::set Object::qualifiedDataNames() const { std::set qualifiedNames = name.empty() || util::contains(name.str(), '.') ? std::set{} : std::set{name}; for (std::shared_ptr const& subObjectNode: subObjects) { yulAssert(qualifiedNames.count(subObjectNode->name) == 0, ""); if (util::contains(subObjectNode->name.str(), '.')) continue; qualifiedNames.insert(subObjectNode->name); if (auto const* subObject = dynamic_cast(subObjectNode.get())) for (YulString const& subSubObj: subObject->qualifiedDataNames()) if (subObject->name != subSubObj) { yulAssert(qualifiedNames.count(YulString{subObject->name.str() + "." + subSubObj.str()}) == 0, ""); qualifiedNames.insert(YulString{subObject->name.str() + "." + subSubObj.str()}); } } yulAssert(qualifiedNames.count(YulString{}) == 0, ""); qualifiedNames.erase(YulString{}); return qualifiedNames; } std::vector Object::pathToSubObject(YulString _qualifiedName) const { yulAssert(_qualifiedName != name, ""); yulAssert(subIndexByName.count(name) == 0, ""); if (boost::algorithm::starts_with(_qualifiedName.str(), name.str() + ".")) _qualifiedName = YulString{_qualifiedName.str().substr(name.str().length() + 1)}; yulAssert(!_qualifiedName.empty(), ""); std::vector subObjectPathComponents; boost::algorithm::split(subObjectPathComponents, _qualifiedName.str(), boost::is_any_of(".")); std::vector path; Object const* object = this; for (std::string const& currentSubObjectName: subObjectPathComponents) { yulAssert(!currentSubObjectName.empty(), ""); auto subIndexIt = object->subIndexByName.find(YulString{currentSubObjectName}); yulAssert( subIndexIt != object->subIndexByName.end(), "Assembly object <" + _qualifiedName.str() + "> not found or does not contain code." ); object = dynamic_cast(object->subObjects[subIndexIt->second].get()); yulAssert(object, "Assembly object <" + _qualifiedName.str() + "> not found or does not contain code."); yulAssert(object->subId != std::numeric_limits::max(), ""); path.push_back({object->subId}); } return path; } shared_ptr Object::objectAt(shared_ptr const& _object, string const& _qualifiedName) { if (_qualifiedName.empty() || _qualifiedName == _object->name.str()) return _object; if (!boost::algorithm::starts_with(_qualifiedName, _object->name.str() + ".")) return nullptr; string const subObjectPath = _qualifiedName.substr(_object->name.str().length() + 1); string const subObjectName = subObjectPath.substr(0, subObjectPath.find_first_of('.')); auto subObjectIt = ranges::find_if( _object->subObjects, [&subObjectName](auto const& _subObject) { return _subObject->name.str() == subObjectName; } ); if (subObjectIt == _object->subObjects.end()) return nullptr; auto subObject = dynamic_pointer_cast(*subObjectIt); yulAssert(subObject, "Assembly object <" + subObject->name.str() + "> does not contain code."); return objectAt(subObject, subObjectPath); }