-
Notifications
You must be signed in to change notification settings - Fork 6k
New issue
Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.
By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.
Already on GitHub? Sign in to your account
Introduce universal compilation host to execution framework #15960
base: develop
Are you sure you want to change the base?
Conversation
Thank you for your contribution to the Solidity compiler! A team member will follow up shortly. If you haven't read our contributing guidelines and our review checklist before, please do it now, this makes the reviewing process and accepting your contribution smoother. If you have any questions or need our help, feel free to post them in the PR or talk to us directly on the #solidity-dev channel on Matrix. |
08ce0db
to
9387714
Compare
6a47a21
to
9e50f8c
Compare
There was an error when running
Please check that your changes are working as intended. |
f79fbae
to
3b47946
Compare
9e5df30
to
eb50f43
Compare
eb50f43
to
132c6f4
Compare
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Some things need to be ironed out still, on a small scale you should make a pass over the PR and fix all the auto
s to use auto const&
and auto const*
where appropriate. Or respective mutable reference/pointer types. Or nothing, if you explicitly want to make a copy :)
Also (but optionally), I personally like to make as much const as possible. For ex., if you have a bool shouldDoSomething = m_compiler.shouldDoSomething()
will the sole purpose to pass it on and not modify it further, I declare it as bool const shouldDoSomething = ...
.
On the bigger scale of things, I am not a big fan of these optionals everywhere to capture experimental solidity in which some data members aren't available. It leads to a lot of bloat. Perhaps a variant with two clean distinct structs would be better. But not sure, really.
) const | ||
{ | ||
std::vector<ContractDefinition const*> contracts; | ||
if (auto source = m_sources.find(_sourceName); source != m_sources.end()) |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
if (auto source = m_sources.find(_sourceName); source != m_sources.end()) | |
if ( | |
auto const& source = m_sources.find(_sourceName); | |
source != m_sources.end() | |
) |
aside from that, do you think it would make sense to assert that there actually is a contract definition with the source name instead of silently returning an empty vector?
libsolidity/util/compiler/Compiler.h | ||
libsolidity/util/compiler/Compiler.cpp | ||
libsolidity/util/compiler/InternalCompiler.h | ||
libsolidity/util/compiler/InternalCompiler.cpp | ||
libsolidity/util/compiler/CompilerHost.h | ||
libsolidity/util/compiler/CompilerHost.cpp |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
can you order this alphabetically? :)
@@ -38,7 +38,8 @@ namespace solidity::frontend::test | |||
#define CHECK_DEPLOY_GAS(_gasNoOpt, _gasOpt, _evmVersion) \ | |||
do \ | |||
{ \ | |||
u256 metaCost = GasMeter::dataGas(m_compiler.cborMetadata(m_compiler.lastContractName()), true, _evmVersion); \ | |||
auto compiledContract = m_compiler.output().contract().value(); \ |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
this unnecessarily copies around compile contract instances. I see you implemented a signature
std::optional<CompiledContract> contract(ContractName const& _name = {}) const;
Instead, I would implement it as CompiledContract const* contract(ContractName const& _name = {}) const;
. Ultimately the compiled contract comes from the m_sourceUnits
map so all you have to do is point to it if it exists, return nullptr
otherwise. Then you can also write the following
auto compiledContract = m_compiler.output().contract().value(); \ | |
auto const* compiledContract = m_compiler.output().contract().value(); \ |
@@ -202,8 +203,9 @@ BOOST_AUTO_TEST_CASE(single_callvaluecheck) | |||
} | |||
)"; | |||
compileAndRun(sourceCode); | |||
size_t bytecodeSizeNonpayable = m_compiler.object("Nonpayable").bytecode.size(); | |||
size_t bytecodeSizePayable = m_compiler.object("Payable").bytecode.size(); | |||
auto compilerOutput = m_compiler.output(); |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
auto compilerOutput = m_compiler.output(); | |
auto const& compilerOutput = m_compiler.output(); |
the way you did it will make a copy of the output
size_t bytecodeSizeNonpayable = compilerOutput.contract("Nonpayable").value().object.size(); | ||
size_t bytecodeSizePayable = compilerOutput.contract("Payable").value().object.size(); |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
this does not assert that "Nonpayable"
and "Payable"
even exist as far as i can tell
for (auto sourceName: m_stack.sourceNames()) | ||
{ | ||
std::vector<CompiledContract> compiled; | ||
for (auto contract: m_stack.contractDefinitions(sourceName)) |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
for (auto contract: m_stack.contractDefinitions(sourceName)) | |
for (auto const& contract: m_stack.contractDefinitions(sourceName)) |
std::string contractName = contract->fullyQualifiedName(); | ||
LinkerObject object = m_stack.object(contractName); | ||
LinkerObject runtimeObject = m_stack.runtimeObject(contractName); | ||
bool hasUnlinkedReferences = !object.linkReferences.empty(); |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
this is unused
for (auto const& error: m_stack.errors()) | ||
{ | ||
auto formatted = SourceReferenceFormatter::formatErrorInformation( | ||
*error.get(), |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
*error.get(), | |
*error, |
std::vector<AnnotatedEventSignature> signatures; | ||
ContractDefinition const& contract = m_stack.contractDefinition(_contractName); | ||
|
||
for (EventDefinition const* event: contract.events() + contract.usedInterfaceEvents()) |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
you can use ranges::views::concat
here to avoid creating an entirely new vector
#pragma once | ||
|
||
#include <liblangutil/SourceReferenceFormatter.h> | ||
#include <libsolutil/Common.h> |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Does this use anything from this header?
Description
This PR implements a universal compiler host for running tests with various compilers (etc. the internal compiler aka.
CompilerStack
or standard JSON compiler).Details
It introduces some common types for the compiler inputs and its output. A compilation module that wraps the linked / internal compiler was added. It can be used in a newly added universal compiler host. This host is then used inside the test execution framework.
Outlook
The input / output types introduced by this PR were inspired by the
SolidityCompilationFramework
. In a future PR the universal compiler host could be used there as well.Ultimately, this PR makes it possible to implement a compilation module for the standard JSON compiler as well and use that in the pipeline.