-
Notifications
You must be signed in to change notification settings - Fork 15
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
feat: storage layout + read/write slots + mut keyword #30
Conversation
name: String, | ||
symbol: String, | ||
decimals: u8, | ||
total_supply: Slot<U256>, |
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.
We chatted about some of this offline, I think it's a bit odd to have Slot
here but not in Mapping, but good for a first version and can be improved in the future.
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.
So if I understand correctly, every type used in a struct that derives storage
has to implement StorageLayout
, right?
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.
We chatted about some of this offline, I think it's a bit odd to have
Slot
here but not in Mapping, but good for a first version and can be improved in the future.
yes, my bad. I totally forgot... I'll amend the PR to incorporate that design pattern.
So if I understand correctly, every type used in a struct that derives
storage
has to implementStorageLayout
, right?
yes, that's correct they need to impl StorageLayout
so that we can ensure that we can allocate a slot in the layout for them.
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.
I think this is fine as is and we can iterate later on
pub fn total_supply(&self) -> U256 { | ||
self.total_supply | ||
self.total_supply.read() |
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.
I think you wrote this somewhere, but pub
fields could also get automatic getters, can be done in the future. (let's open an issue)
fn call(&self); | ||
fn call_with_data(&self, calldata: &[u8]); | ||
fn call(&mut self); | ||
fn call_with_data(&mut self, calldata: &[u8]); |
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.
It could also make sense later on to have immutable versions of these, not sure how that would work but it'd be cool.
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.
agreed.
do you foresee any scenario other than static calls where we would use the not mutable call?
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.
not right now
padded[..bytes.len()].copy_from_slice(&bytes); | ||
sstore(key, U256::from_be_bytes(padded)); | ||
impl<K, V> StorageLayout for Mapping<K, V> { | ||
fn allocate(slot: u64) -> Self { |
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.
Nice!
eth-riscv-runtime/src/types/slot.rs
Outdated
/// Wrapper around `alloy::primitives` that can be written in a single slot (single EVM word). | ||
#[derive(Default)] | ||
pub struct Slot<V> { | ||
id: u64, |
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.
Technically, shouldn't the id be u256? I agree it should be fine as u64 if we incrementally allocate slots, but to be fully compatible/flexible and allow custom storage layouts from the user I think this would have to be u256?
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.
yes, it should be u256, i just kept the u64 as that's what was being used in Mapping
. I'll do a follow-up commit to make everything compliant.
contract-derive/src/lib.rs
Outdated
}); | ||
|
||
// Generate initialization code for each field | ||
let init_fields = fields.iter().enumerate().map(|(slot, f)| { |
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.
I think this is fine for now, but we should spend some time crafting how storage layout will work. For example, this won't allow for storing a (u256, u256)
sequentially, we'd need to implement StorageLayout
for it with a phantom slot, and then in the actual read/write use some encoded key (similar to Mapping). Ideally we'd know the storage size of the type at compile time (if value type) and use that size for allocation instead of enumerate
. Reference types (Mapping
, arrays, etc) could work like Mapping. This is what Solidity does also.
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 a really good point and i totally missed that. Do you want me to try to address it in this PR? or would u open an issue and do it in the following one?
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.
I think later is fine
* layout with U256 slots * adjust details for EVM compliance * finish tests
@leonardoalt changes since your review:
finally, i tracked the todo's here: |
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.
Just one small comment
README.md
Outdated
@@ -37,7 +37,7 @@ out-of-the-box. | |||
use core::default::Default; | |||
|
|||
use contract_derive::contract; | |||
use eth_riscv_runtime::types::Mapping; | |||
use eth_riscv_runtime::storage::Mapping; |
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.
I think Mapping
should still be importable from something more generic like types
, wdyt?
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.
sure, i just thought storage
would be more informative.
we can move back to a more generic types
_pd: PhantomData<(K, V)>, | ||
} | ||
|
||
impl<K, V> StorageLayout for Mapping<K, V> { | ||
fn allocate(slot: u64) -> Self { | ||
fn allocate(first: u64, second: u64, third: u64, fourth: u64) -> Self { |
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.
Since we're always allocating U256 slots, could change this to U256 as well?
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.
unfortunately U256 doesn't impl ToTokens
and i was getting compiler errors, so i had to deconstruct it into limbs and then ensamble it again...
maybe i could do a PR to alloy or something though
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.
otherwise i can maybe add comments for it to make it more comprehensive
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.
ah right
eth-riscv-runtime/src/storage/mod.rs
Outdated
pub trait StorageLayout { | ||
fn allocate(slot: u64) -> Self; | ||
fn allocate(limb0: u64, limb1: u64, limb2: u64, limb3: u64) -> Self; |
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.
single U256?
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.
same reason as above
mut
keyword for write operations, so that the compiler complains when a contract fn tries to update storage without&mut self
Slot<V>
for primitives that fit a single storage slot (equivalent to the oldWord<V>
>>> as discussed, we don't like this wrapper type, so i think that down the line we can enhance thecontract
macro so that the devex is like working with primitives.storage
that initializes the storage layout >>> it requires all the attributes of the struct to implement theStorageLayout
trait (some notes regarding this point):StorageLayout
trait is unnecessary and it could be merged with the existingStorageStorable
.u64
toOption<u64>
or something like this.PS: i know i'm not the best at coming up with good names, so feel free to propose new ones