[libevmasm] Remove m_sourceList member from Assembly class.

This commit is contained in:
Alexander Arlt 2023-04-26 19:41:45 -05:00
parent 7cfb34ce76
commit 98944cba5b
2 changed files with 30 additions and 35 deletions

View File

@ -75,19 +75,19 @@ unsigned Assembly::codeSize(unsigned subTagSize) const
} }
} }
void Assembly::importAssemblyItemsFromJSON(Json::Value const& _code) void Assembly::importAssemblyItemsFromJSON(Json::Value const& _code, vector<string> const& _sourceList)
{ {
solAssert(m_items.empty()); solAssert(m_items.empty());
solRequire(_code.isArray(), AssemblyImportException, "Supplied JSON is not an array."); solRequire(_code.isArray(), AssemblyImportException, "Supplied JSON is not an array.");
for (auto current = begin(_code); current != end(_code); ++current) for (auto current = begin(_code); current != end(_code); ++current)
{ {
auto const& item = m_items.emplace_back(createAssemblyItemFromJSON(*current, m_sourceList)); auto const& item = m_items.emplace_back(createAssemblyItemFromJSON(*current, _sourceList));
if (item == Instruction::JUMPDEST) if (item == Instruction::JUMPDEST)
solThrow(AssemblyImportException, "JUMPDEST instruction without a tag"); solThrow(AssemblyImportException, "JUMPDEST instruction without a tag");
else if (item.type() == AssemblyItemType::Tag) else if (item.type() == AssemblyItemType::Tag)
{ {
++current; ++current;
if (current != end(_code) && createAssemblyItemFromJSON(*current, m_sourceList) != Instruction::JUMPDEST) if (current != end(_code) && createAssemblyItemFromJSON(*current, _sourceList) != Instruction::JUMPDEST)
solThrow(AssemblyImportException, "JUMPDEST expected after tag."); solThrow(AssemblyImportException, "JUMPDEST expected after tag.");
} }
} }
@ -424,7 +424,7 @@ std::string Assembly::assemblyString(
return tmp.str(); return tmp.str();
} }
Json::Value Assembly::assemblyJSON(bool _includeSourceList) const Json::Value Assembly::assemblyJSON(std::vector<std::string> const& _sources, bool _includeSourceList) const
{ {
Json::Value root; Json::Value root;
root[".code"] = Json::arrayValue; root[".code"] = Json::arrayValue;
@ -433,10 +433,11 @@ Json::Value Assembly::assemblyJSON(bool _includeSourceList) const
{ {
int sourceIndex = -1; int sourceIndex = -1;
if (item.location().sourceName) if (item.location().sourceName)
for (size_t index = 0; index < _sources.size(); ++index)
if (_sources[index] == *item.location().sourceName)
{ {
for (size_t index = 0; index < m_sourceList.size(); ++index)
if (m_sourceList[index] == *item.location().sourceName)
sourceIndex = static_cast<int>(index); sourceIndex = static_cast<int>(index);
break;
} }
auto [name, data] = item.nameAndData(m_evmVersion); auto [name, data] = item.nameAndData(m_evmVersion);
@ -470,12 +471,12 @@ Json::Value Assembly::assemblyJSON(bool _includeSourceList) const
code.append(std::move(jumpdest)); code.append(std::move(jumpdest));
} }
} }
if (_includeSourceList) if (!_sources.empty() && _includeSourceList)
{ {
root["sourceList"] = Json::arrayValue; root["sourceList"] = Json::arrayValue;
Json::Value& jsonSourceList = root["sourceList"]; Json::Value& jsonSourceList = root["sourceList"];
for (int index = 0; index < static_cast<int>(m_sourceList.size()); ++index) for (int index = 0; index < static_cast<int>(_sources.size()); ++index)
jsonSourceList[index] = m_sourceList[static_cast<size_t>(index)]; jsonSourceList[index] = _sources[static_cast<size_t>(index)];
} }
if (!m_data.empty() || !m_subs.empty()) if (!m_data.empty() || !m_subs.empty())
@ -490,8 +491,7 @@ Json::Value Assembly::assemblyJSON(bool _includeSourceList) const
{ {
stringstream hexStr; stringstream hexStr;
hexStr << hex << i; hexStr << hex << i;
m_subs[i]->setSourceList(m_sourceList); data[hexStr.str()] = m_subs[i]->assemblyJSON(_sources, false);
data[hexStr.str()] = m_subs[i]->assemblyJSON(false);
} }
} }
@ -501,7 +501,7 @@ Json::Value Assembly::assemblyJSON(bool _includeSourceList) const
return root; return root;
} }
shared_ptr<Assembly> Assembly::fromJSON(Json::Value const& _json, vector<string> const& _sourceList, int _level) std::pair<std::shared_ptr<Assembly>, std::vector<std::string>> Assembly::fromJSON(Json::Value const& _json, vector<string> const& _sourceList, int _level)
{ {
solRequire(_json.isObject(), AssemblyImportException, "Supplied JSON is not an object."); solRequire(_json.isObject(), AssemblyImportException, "Supplied JSON is not an object.");
static set<string> const validMembers{".code", ".data", ".auxdata", "sourceList"}; static set<string> const validMembers{".code", ".data", ".auxdata", "sourceList"};
@ -528,15 +528,23 @@ shared_ptr<Assembly> Assembly::fromJSON(Json::Value const& _json, vector<string>
); );
shared_ptr<Assembly> result = make_shared<Assembly>(langutil::EVMVersion(), _level == 0, ""); shared_ptr<Assembly> result = make_shared<Assembly>(langutil::EVMVersion(), _level == 0, "");
vector<string> sourceList;
if (_json.isMember("sourceList")) if (_json.isMember("sourceList"))
{ {
solAssert(_level == 0); solAssert(_level == 0);
for (auto const& it: _json["sourceList"]) for (auto const& it: _json["sourceList"])
result->m_sourceList.emplace_back(it.asString()); {
solRequire(
std::find(sourceList.begin(), sourceList.end(), it.asString()) == sourceList.end(),
AssemblyImportException,
"Items in 'sourceList' array are not unique."
);
sourceList.emplace_back(it.asString());
}
} else } else
result->m_sourceList = _sourceList; sourceList = _sourceList;
result->importAssemblyItemsFromJSON(_json[".code"]); result->importAssemblyItemsFromJSON(_json[".code"], sourceList);
if (_json[".auxdata"]) if (_json[".auxdata"])
{ {
solRequire(_json[".auxdata"].isString(), AssemblyImportException, "Optional member '.auxdata' is not of type string."); solRequire(_json[".auxdata"].isString(), AssemblyImportException, "Optional member '.auxdata' is not of type string.");
@ -565,7 +573,7 @@ shared_ptr<Assembly> Assembly::fromJSON(Json::Value const& _json, vector<string>
} }
else if (code.isObject()) else if (code.isObject())
{ {
shared_ptr<Assembly> subassembly(Assembly::fromJSON(code, result->m_sourceList, _level + 1)); shared_ptr<Assembly> subassembly(Assembly::fromJSON(code, sourceList, _level + 1).first);
solAssert(subassembly); solAssert(subassembly);
result->m_subs.emplace_back(make_shared<Assembly>(*subassembly)); result->m_subs.emplace_back(make_shared<Assembly>(*subassembly));
// TODO: this shouldn't be enough for the general case. // TODO: this shouldn't be enough for the general case.
@ -575,7 +583,7 @@ shared_ptr<Assembly> Assembly::fromJSON(Json::Value const& _json, vector<string>
solThrow(AssemblyImportException, "Key inside '.data' '" + dataItemID + "' can only be a valid hex-string or an object."); solThrow(AssemblyImportException, "Key inside '.data' '" + dataItemID + "' can only be a valid hex-string or an object.");
} }
} }
return result; return std::make_pair(result, sourceList);
} }
AssemblyItem Assembly::namedTag(string const& _name, size_t _params, size_t _returns, optional<uint64_t> _sourceID) AssemblyItem Assembly::namedTag(string const& _name, size_t _params, size_t _returns, optional<uint64_t> _sourceID)

View File

@ -150,31 +150,19 @@ public:
) const; ) const;
/// Create a JSON representation of the assembly. /// Create a JSON representation of the assembly.
Json::Value assemblyJSON( Json::Value assemblyJSON(std::vector<std::string> const& _sources, bool _includeSourceList = true) const;
bool _includeSourceList = true
) const;
/// Loads the JSON representation of assembly. /// Loads the JSON representation of assembly.
/// @param _json JSON object containing assembly in the format produced by assemblyJSON(). /// @param _json JSON object containing assembly in the format produced by assemblyJSON().
/// @param _sourceList list of source names (used to supply the source-list from the root-assembly object to sub-assemblies). /// @param _sourceList list of source names (used to supply the source-list from the root-assembly object to sub-assemblies).
/// @param _level this function might be called recursively, _level reflects the nesting level. /// @param _level this function might be called recursively, _level reflects the nesting level.
/// @returns Resulting Assembly object loaded from given json. /// @returns Resulting Assembly object loaded from given json.
static std::shared_ptr<Assembly> fromJSON( static std::pair<std::shared_ptr<Assembly>, std::vector<std::string>> fromJSON(
Json::Value const& _json, Json::Value const& _json,
std::vector<std::string> const& _sourceList = {}, std::vector<std::string> const& _sourceList = {},
int _level = 0 int _level = 0
); );
void setSourceList(std::vector<std::string> const& _sourceList)
{
m_sourceList = _sourceList;
}
std::vector<std::string> const& sourceList()
{
return m_sourceList;
}
/// Mark this assembly as invalid. Calling ``assemble`` on it will throw. /// Mark this assembly as invalid. Calling ``assemble`` on it will throw.
void markAsInvalid() { m_invalid = true; } void markAsInvalid() { m_invalid = true; }
@ -195,7 +183,7 @@ protected:
/// the code array. This method only works on clean Assembly objects that don't have any items defined yet. /// the code array. This method only works on clean Assembly objects that don't have any items defined yet.
/// @param _json JSON array that contains assembly items (e.g. json['.code']) /// @param _json JSON array that contains assembly items (e.g. json['.code'])
/// @param _sourceList list of source names. /// @param _sourceList list of source names.
void importAssemblyItemsFromJSON(Json::Value const& _code); void importAssemblyItemsFromJSON(Json::Value const& _code, std::vector<std::string> const& _sourceList);
/// Creates an AssemblyItem from a given JSON representation. /// Creates an AssemblyItem from a given JSON representation.
/// @param _json JSON object that consists a single assembly item /// @param _json JSON object that consists a single assembly item
@ -249,7 +237,6 @@ protected:
/// Internal name of the assembly object, only used with the Yul backend /// Internal name of the assembly object, only used with the Yul backend
/// currently /// currently
std::string m_name; std::string m_name;
std::vector<std::string> m_sourceList;
langutil::SourceLocation m_currentSourceLocation; langutil::SourceLocation m_currentSourceLocation;
public: public: