Skip to content
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

Fix ternary tests and optimize TERNARY macro #97

Merged
merged 4 commits into from
Feb 20, 2023

Conversation

d1ll0n
Copy link
Contributor

@d1ll0n d1ll0n commented Feb 17, 2023

Old TERNARY macro:

// Input Stack: [condition, x, y]

// If the condition is false, jump and pop
iszero __Utils_Misc__ternaryNoSwap jumpi
    swap1 // [y, x]
__Utils_Misc__ternaryNoSwap:
   // condition ? [y, x] : [x, y]
    pop // condition ? [x] : [y]

When the condition is true, the operations are:
iszero push jumpi swap jumpdest pop = 23 gas
When it's false:
iszero push jumpi jumpdest pop = 20 gas

If we use branchless logic we can get this down to 17 for both cases:


// Input Stack: [condition, x, y]
swap1           // [x, condition, y]
dup3            // [y, x, condition, y]
xor mul xor     // [((x ^ y) * condition) ^ y]
// Output Stack: [condition ? x : y]

swap1 dup3 xor mul xor = 17 gas

I don't think it's possible to do the same optimization for the inverse macro NOT_TERNARY because you would either need to swap x and y in the formula, adding an additional swap operation, or use iszero. Since that macro is already 3 gas less than the TERNARY macro, adding another operation to the branchless version makes it equivalent for one case and more expensive for the other.

For reference, there are two formulae you can use:

// condition ? x : y

(x - y) * condition + y
// Required Stack: [x, y, condition, y]
// sub mul add

((x ^ y) * condition) ^ y
// Required Stack: [x, y, condition, y] OR [y, x, condition, y]
// xor mul xor

And for the inverse

// condition ? y : x

(y - x) * condition + x
// Required Stack: [y, x, condition, x]
// sub mul add

((x ^ y) * condition) ^ x
// Required Stack: [x, y, condition, x] OR [y, x, condition, x]
// xor mul xor

@devtooligan devtooligan changed the base branch from main to dev February 20, 2023 18:23
Copy link
Collaborator

@devtooligan devtooligan left a comment

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

lgtm

@devtooligan devtooligan merged commit b6cac47 into huff-language:dev Feb 20, 2023
Sign up for free to join this conversation on GitHub. Already have an account? Sign in to comment
Labels
None yet
Projects
None yet
Development

Successfully merging this pull request may close these issues.

2 participants