-
Notifications
You must be signed in to change notification settings - Fork 39
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
Packing balance and accruedSince in the same slot #111
Comments
The way streams / flows work looks confusing, I'm assuming it's not defined yet. struct UbiStream {
address to;
uint32 streamAmount;
}
struct UbiAccount {
uint152 balance;
uint32 accruedSince; // timestamp of last redeem
uint32 accrualPerSecond; // if you're not a human, you can still be streamed ubi.
bool isHuman; // allows to NOT require a poh check on transfer, which saves 2600 gas.
// when the third party disables the isHuman and streams, it gets all the UBI the human didn't redeem.
// even the UBI that is streamed to the disabled human (for technical reasons).
uint32 streamedToOthers; // sum of ubi that this human is streaming to others.
Stream[] streamsTo; // purposely not a mapping. allows a third party to disable the streams
// on everyone that has been awarded an ubi stream from this party.
// (should be slightly cheap since this array will be set to zero as it iterates, refunding gas)
} note that im considering that every human is able to do multiple streams but I think that's a bad idea. I would just have each human either not streaming, or streaming everything to another human. That makes a few steps easier, and removes computation (that also results in less gas costs for everyone) so, this is what id do. It's will be cheaper and way easier to interact with in the edge cases (revoking streams, awarding streams, updating human status of an account...) struct UbiAccount {
uint176 balance;
uint32 accruedSince; // timestamp of last redeem.
uint32 nStreams; // counts how many ubi streams are pointing at you.
bool isHuman; // allows to NOT require a poh check on transfer, which saves 2600 gas.
// when a third party disables the isHuman, it gets all the accrued UBI of one stream.
// but the UBI that was streamed from other to this human will stay in the human.
bool streamingToOther; // makes transfer cheaper, otherwise you'd have to SLOAD the "streamsTo"
// address of this struct, that lives in another slot.
// --- 2nd slot ---
address streamsTo; // if streamingToOther, a whole human stream is awarded to this address.
uint96 freespace;
} |
For the first comment about the improvement: For the second about delegations:
The UBI Delegation pattern is based on a "Plugin" system where the delegation contract should implement the |
instead of having:
you can do this:
and update everything to support this new structure (cast uint256s to uint224s, etc)
Rationale
Every time you transfer, you need to write to both slots. Assuming best case (non-zero to non-zero), the
updateBalance(...)
that is called will incur:(2900 + 2100) * 2 = 10_000
if you however have both the balance and accruedSince in the same slot, you rewrite on dirty and hot:
(2900 + 2100) + (100 + 100) = 5_200
all in all, this amounts to 4_800 gas saved.
The text was updated successfully, but these errors were encountered: