-
Notifications
You must be signed in to change notification settings - Fork 12
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: added support for comparison operators #86
base: main
Are you sure you want to change the base?
Conversation
We still need to add way more testing to this. also need to address the changes from the PR with the edge case bugs. |
let carry_shift = 0x1000000000000000000000000000000; | ||
|
||
let mut addend: [Field; N] = [0; N]; | ||
if comp_result == true { |
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 was first thinking about skipping the if statement by casting the bool
to Field
and then computing intended_lhs = (1- comp_result as Field)* lhs + comp_result * rhs
but not sure whether looping over the limbs and doing field multiplications would be any better than a conditional.
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.
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 you just want to swap the lhs and rhs then you can do
let (lhs, rhs) = if comp_result {
(rhs, lhs)
} else {
(lhs, rhs)
}
This would end up with needing to merge two [Field;N]
values but that's more efficient than duplicating the logic currently in the if-statement. (This could be made better by just returning the element-wise difference between the two arrays rather than directly reordering 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.
The best thing to do in this situation however would be to integrate the benchmarking system I added to https://github.com/noir-lang/noir-library-starter.
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.
thanks I was wondering if there's a way to do a swap in a single line :D
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 still need to reduce this copy-pasting.
src/fns/unconstrained_helpers.nr
Outdated
let underflow = b_u60.gte(a_u60); | ||
|
||
if underflow { | ||
temp_u60 = a_u60; |
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.
wonder how bad this is in brillig @TomAFrench. should I do what I did in the constrained version and just duplicate the code?
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.
Could you not refactor this function to have a private implementation which accepts two U60Repr
s?
pub(crate) unconstrained fn __cmp_remainder<let N: u32>(
lhs: [Field; N],
rhs: [Field; N],
) -> (bool, [Field; N], [bool; N], [bool; N]) {
let mut a_u60: U60Repr<N, 2> = U60Repr::from(lhs);
let mut b_u60: U60Repr<N, 2> = U60Repr::from(rhs);
let mut temp_u60: U60Repr<N, 2> = U60Repr::one();
let underflow = b_u60.gte(a_u60);
if underflow {
__cmp_remainder_priv(b_u60, a_u60);
} else {
__cmp_remainder_priv(a_u60, b_u60);
}
Again, benchmarking would be way to make these decisions.
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.
yep I can make a private function that does the second part (computing carries etc). will do that now.
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.
did the refactoring. I think we can refactor it a lot here. these operations are done over and over in all the with_flags
functions.
@@ -294,6 +294,58 @@ pub(crate) fn validate_gt<let N: u32, let MOD_BITS: u32>(lhs: [Field; N], rhs: [ | |||
assert(result_limb == 0); | |||
} | |||
|
|||
pub(crate) fn cmp<let N: u32, let MOD_BITS: u32>(lhs: [Field; N], rhs: [Field; N]) -> bool { |
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.
Why not return an Ordering
from this function?
let borrow_shift = 0x1000000000000000000000000000000; | ||
let carry_shift = 0x1000000000000000000000000000000; |
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.
These are the same value so why not have a single constant? Also this should be named to make it clear how big they are.
let carry_shift = 0x1000000000000000000000000000000; | ||
|
||
let mut addend: [Field; N] = [0; N]; | ||
if comp_result == true { |
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 still need to reduce this copy-pasting.
Description
added support for comparison operators between bignums
Problem*
Resolves
Summary*
Additional Context
PR Checklist*
cargo fmt
on default settings.