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

Dead Instruction Elimination pass removes out of bounds array sets #5296

Closed
Thunkar opened this issue Jun 20, 2024 · 6 comments
Closed

Dead Instruction Elimination pass removes out of bounds array sets #5296

Thunkar opened this issue Jun 20, 2024 · 6 comments
Assignees
Labels
bug Something isn't working

Comments

@Thunkar
Copy link
Contributor

Thunkar commented Jun 20, 2024

Aim

While moving some tests from constrained to unconstrained so they run faster (we're only checking function logic and assume circuit generation is fine), the code that used to work started failing with "Array index out of bounds" errors.

Expected Behavior

Behavior should be consistent between constrained and unconstrained contexts.

Bug

Out of bounds errors can be masked in certain scenarios, specifically when copying array positions from one to another. However, illegal acceses are always detected as expected in unconstrained functions.

To Reproduce

// Taken from noir-protocol-circuits
fn arr_copy_slice<T, N, M>(src: [T; N], mut dst: [T; M], offset: u32) -> [T; M] {
    for i in 0..dst.len() {
        // We are illegally accessing the src array here if dst.len() > src.len()
        dst[i] = src[i + offset];
    }
    dst
}

unconstrained fn boom<N, M>(src: [Field; N], dst: [Field; M]) -> [Field; M] {
    arr_copy_slice(src, dst, 0)
}


fn main() {
    // This shouldn't work
    let small_array_src = [1; 5];
    let mut big_array_dst = [0; 6];
    let mut result = arr_copy_slice(small_array_src, big_array_dst, 0);

    // Uncomment for fireworks. This *always* get caught in unconstrained land
    //let result = boom();

    // This is fine
    let first = result[0];
    dep::std::println(first);
    // This is also fine, since we are overwriting the "illegal" position.
    // BUT if we comment this line out, fireworks again (since we are now "using" the illegal position)
    result[5] = 1;

    dep::std::println(result[5]);
}

Project Impact

Nice-to-have

Impact Context

No response

Workaround

None

Workaround Description

No response

Additional Context

No response

Installation Method

Compiled from source

Nargo Version

No response

NoirJS Version

No response

Would you like to submit a PR for this Issue?

None

Support Needs

No response

@Thunkar Thunkar added the bug Something isn't working label Jun 20, 2024
@github-project-automation github-project-automation bot moved this to 📋 Backlog in Noir Jun 20, 2024
@jfecher
Copy link
Contributor

jfecher commented Jun 20, 2024

Interesting, it looks like the pass to remove unused code removes the array_set instruction in question since it was unused in the final program. This explains why it also still catches it if that index is later printed out without being mutated first.

@jfecher jfecher changed the title Illegal array accesses in constrained not detected Dead Instruction Elimination pass removes out of bounds array sets Jun 20, 2024
@asterite
Copy link
Collaborator

Reduced:

fn main() {
    let src = [1];
    let mut dst = [1];
    dst[0] = src[10];
    dst[0] = 1; // commenting this line will produce Index Out of Bounds when executing
    println(dst[0])
}

@asterite
Copy link
Collaborator

One solution could be to not remove array_get operations if the index might be out of bounds (if the index is a constant and it's out of bounds, or if the index is not known at compile-time)

@asterite
Copy link
Collaborator

This program also doesn't give index out of bounds at runtime, but probably should:

fn main() {
    let mut dst = [1];
    dst[10] = 1;
}

@vezenovm
Copy link
Contributor

vezenovm commented Aug 6, 2024

Posting discussion from scrum for visibility:

We are going to try a new approach. Instead of always inserting a length assertion on array gets/sets, when we remove an unused array operation during DIE, we will replace it with a length assertion. This should hopefully let us avoid any extra constraints on dynamic array gets/sets which are still used elsewhere in the circuit.

github-merge-queue bot pushed a commit that referenced this issue Aug 13, 2024
…unds (#5691)

# Description

## Problem

For #5296

## Summary

In DIE, when we find an `ArrayGet/Set` that we could remove, if that
instruction might be an index out of bounds we still remove it but leave
a constrain check too.

## Additional Context

For now the code is a bit messy, I just wanted to get it working first.

Also, it's not working for slices yet because we can't insert out of
bounds checks if we don't know the slice length. For that we'll need to
add the checks like we do for `ArrayGet` for slice (this could be done
in a separate PR, though).

## Documentation\*

Check one:
- [x] No documentation needed.
- [ ] Documentation included in this PR.
- [ ] **[For Experimental Features]** Documentation to be submitted in a
separate PR.

# PR Checklist\*

- [x] I have tested the changes locally.
- [x] I have formatted the changes with [Prettier](https://prettier.io/)
and/or `cargo fmt` on default settings.
@vezenovm
Copy link
Contributor

Closed due to #5691 being merged

@github-project-automation github-project-automation bot moved this from 📋 Backlog to ✅ Done in Noir Aug 16, 2024
Sign up for free to join this conversation on GitHub. Already have an account? Sign in to comment
Labels
bug Something isn't working
Projects
Archived in project
4 participants