-
Notifications
You must be signed in to change notification settings - Fork 287
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
Info regarding std
and no_std
incomplete or missing
#531
Comments
I got stuck trying to sort out exactly the same question. I'm working on some draft documentation for this topic, but I'm going to need some technical assistance because the current doc (which I took a stab at fixing and failed) seems contradictory. @shawntabrizi can you provide some detail around how this works or suggest someone I should reach out to? |
@lisa-parity Which doc exactly do you mean by "the current doc"? |
What I was looking at revising was in the Add a Nicks pallet tutorial (most likely, I would make this its own topic instead of embedding the information inside of the tutorial--that was the "plan" behind the XXX placeholder--oops, my bad!). |
@lisa-parity Making this its own topic seems like a good idea. There seems to be some duplicated content between these tutorials, but many of the more technical details are missing. I found this:
For a while, I thought that this answered all my question. But the more I think about this, the more it confuses me. So in addition to all questions above, I now have this problem: If the Wasm binary is part of the on-chain storage, then it must be loaded during runtime, not during compile time (otherwise, how would forkless upgrades work?). But then why the Wasm binary also included in the native binary at compile-time. And what happens to the included Wasm binary when it becomes outdated due to a forkless upgrade? Funny enough, the dedicated Key Concept page doesn't mention much of this: https://docs.substrate.io/v3/concepts/runtime/ I'm really looking forward to your update :) |
So this is very raw and not technically vetted but this is my best guess so far (and doesn't address your additional questions -- all good points that I don't have answers for): The Substrate runtime compiles to both a native Rust binary that includes standard library (
When you run the
The compiler then builds the native Rust (
DISCLAIMER: This might not make any sense at all. I'm trying to piece together the facts from the clues 😉. |
@lisa-parity Thank you! That's mostly compatible with my understanding. Except (maybe I misunderstood)...
The |
I think this question needs a bit more scope. What you have stated above is an accurate answer for this question. We compile a Wasm runtime first, then use that Wasm to embed into the native client binary. This happens with the
One of the fundamental architecture decisions of Substrate/Polkadot is the use of a Wasm runtime. This Wasm runtime is embedded into the genesis state of your blockchain, and thus must exist to be embeded. This is not optional, but a requirement of the framework. You cannot build the Substrate client without some Wasm file being embeded into the binary. This file can be a dummy file (I think there is a flag for that), but then your node would break/panic if not executing everything in Native. Here is a note about that in Substrate:
For local testing purposes you can build only the native client, skipping a new Wasm build like so:
Here is the note on the environment variable:
In this case,
You can do that with the following script: https://github.com/paritytech/substrate/blob/master/.maintain/build-only-wasm.sh
Yes, it can makes sense to compile only the Wasm binary, if for example you are just trying to provide the upgrade Wasm to perform a forkless upgrade. Practically speaking, usually when you do a runtime upgrade, you want to provide both a native and Wasm binary, so doing only one of those halves usually is an incomplete release process. |
@shawntabrizi Thanks for the answer! Unfortunately, I remain confused. Particularly by this:
First of all, by embedded into the binary you mean that the Wasm binary is compiled into the native binary, like a string literal, right? And by embedded into the genesis state, you mean the Wasm binary is copied into on-chain storage, right? See Architecture. And if so, what's the purpose of having the Wasm binary embedded into the native binary? After the first forkless update (which doesn't change the native binary... right?), the native binary would contain an outdated version of the Wasm binary. That doesn't sound right to me. And in addition to that, what is the purpose of all this? A Wasm binary in on-chain storage for the purpose of forkless upgrades makes sense. But why build the native binary with this strange indirection? |
The native binary at one point ran over 100x faster than the wasm binary, which for performance purposes was significant. People running the latest native client would reap those performance gains when running a node. As of recent, improvements to the wasm execution has made it so the difference is only 10x or less. (still quite large. we hope for this to be equal one day) We even have an issue to remove the native binary from our build process all together: paritytech/substrate#7288 So you are asking the right questions, just something that is still a work in progress.
When starting a new chain you need that initial Wasm binary. The one for Polkadot and Kusama are included in the chain specification found here, (in the genesis state): https://github.com/paritytech/polkadot/tree/master/node/service/res However, what do you think happens when you run a simple dev node starting on block 0? (it uses this embeded wasm) Again, it is not strictly required, as you can skip the wasm build, and configure your node to never look at the wasm binary, however, this is not the greenfield scenario. |
@shawntabrizi Sorry, I'm still not sure I understand.
So for production code, the "default" move is to place the original Wasm binary into on-chain storage at genesis, but a dev node will just use the Wasm embedded into the native binary (no copy)? But this seems strange to me. If production code always refers to on-chain storage for the Wasm, why embed it into the native binary. Why not just build the native binary without embedding the Wasm. I don't know, maybe I'm just being dumb, but this just doesn't check out for me. |
The Wasm is needed to run a dev chain. Like:
In this case, genesis state is generated on the fly, and the Wasm needs to come from somewhere: https://github.com/paritytech/substrate/blob/master/bin/node/cli/src/chain_spec.rs#L298
|
@shawntabrizi Okay, that makes sense. I have two last questions:
Oh, and thanks for bearing with me! |
Not that I know of.
You can, using
|
@shawntabrizi Great! I think that clears everything up. Thank you very much for the all answers! 👍 |
It is better documented now than before. |
Content request
There doesn't seem to be a thorough explanation how exactly
std
andno_std
relate to the Wasm binary,neither in the tutorials nor the docs, and some of the hints on this topic seem very confusing to me. I apologize for the long issue.
In Add a Pallet, we get the following:
It's unfortunate that the link is missing (link to line). Where was this supposed to point?
That's all the info we get from Add a Pallet. The Build a proof of existence dApp tutorial has the following to say:
As a minor complaint, "
no_std
features" sounds wrong:std
is a feature;no_std
is an attribute used in Rust and doesn't really add any functionality. Other than that, the association of Wasm withno_std
sounds ok, until you take a look into the runtime code:The Wasm binary was just associated with
no_std
, why is it now exclusively included ifstd
is enabled? From a stackexchange question, I gather the following: Whencargo build
is called for the runtime, first thebuild.rs
builds a Wasm binary of the runtime (withno_std
), then the runtime is built withstd
enabled and includes the Wasm binary.But this doesn't really answer all of my questions. So I suggest that the following questions be answered/details pointed out:
cargo build
?std
) build, why is the Wasm binary (no_std
) built and included? Why not just do a native build?no_std
Wasm binary? Is there any purpose to doing that?It's quite confusing to me that the Wasm binary, one of the defining features of substrate, is not explained in detail anywhere in the tutorials or the docs. I basically got stuck in the second tutorial trying to understand this.
If you could help me with answers to these questions, I would be perfectly happy to work them into the docs myself.
Are you willing to help with this request?
Maybe (please elaborate above)
The text was updated successfully, but these errors were encountered: