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

Support inferred error sets in recursion #2971

Open
hryx opened this issue Jul 29, 2019 · 3 comments
Open

Support inferred error sets in recursion #2971

hryx opened this issue Jul 29, 2019 · 3 comments
Labels
accepted This proposal is planned. contributor friendly This issue is limited in scope and/or knowledge of Zig internals. proposal This issue suggests modifications. If it also has the "accepted" label then it is planned.
Milestone

Comments

@hryx
Copy link
Contributor

hryx commented Jul 29, 2019

This is a proposal based on the question raised in #763. (If #763 is considered the tracking issue or the final word, please feel free to close this one -- I didn't interpret any of the comments there as conclusive.)

Thanks @rjtobin for resurfacing the issue in IRC.

Summary

If a call graph is recursive, a return type with inferred error set may trigger this compile error:

cannot resolve inferred error set 'glarf': function 'kloyp' not fully analyzed yet

If all functions in that cycle use inferred error sets, the error is guaranteed to happen.

pub fn main() void {
    f1(false) catch {};
}

fn f1(x: bool) !void { // recurses
    return if (x) error.Bad else f1(x);
}

fn f2(x: bool) !void { // recurses
    return if (x) f2(x) else error.Bad;
}

If at least one of the functions in the cycle has an inferred error set, the error may happen. Given this error set:

const E = error{Bad};

Changing f1 to return E!void prevents the error. Leaving f1 alone and instead changing f2 to return E!void does not prevent the error. The outcome may be deterministic or it may be based on the color of the giant spaghetti monster's mood ring, I dunno.

Proposal

Guarantee that inferred error sets are allowed even in cyclic call graphs.

Alternatives

Explicitly state that recursion is incompatible with inferred error sets. "Wait!" I hear you cry, "the documentation already says exactly that". It sure does, but it is also acknowledges that this limitation may be removed in the future. We just need to decide (eventually) whether to add support or officially make it unsupported for the language specification (#75).

Note: Currently, I am neutral on this enhancement. It's not blocking me. This issue is just meant to track it and aggregate potential use cases.

@andrewrk
Copy link
Member

This is implemented and working in self-hosted, only thing left to close the issue is ensure we have behavior test coverage.

@andrewrk andrewrk modified the milestones: 0.10.0, 0.10.1 Aug 24, 2022
@andrewrk andrewrk added the contributor friendly This issue is limited in scope and/or knowledge of Zig internals. label Aug 24, 2022
@andrewrk andrewrk modified the milestones: 0.10.1, 0.11.0 Jan 10, 2023
@andrewrk andrewrk modified the milestones: 0.11.0, 0.12.0 Apr 9, 2023
@mlugg
Copy link
Member

mlugg commented May 5, 2023

Note that this works for plain recursion, but not mutual recursion:

fn f() !void {
    try g(false);
}

fn g(b: bool) !void {
    if (b) {
        try f();
    } else {
        return error.Foo;
    }
}

comptime {
    _ = f; // or g
}
thing.zig:7:14: error: unable to resolve inferred error set
        try f();
            ~^~

@kj4tmp
Copy link
Contributor

kj4tmp commented Oct 1, 2024

This is implemented and working in self-hosted, only thing left to close the issue is ensure we have behavior test coverage.

I started working on this.

I found exactly one test in test/behavior/error.zig:

test "try used in recursive function with inferred error set" {

test "try used in recursive function with inferred error set" {
    if (builtin.zig_backend == .stage2_aarch64) return error.SkipZigTest; // TODO
    if (builtin.zig_backend == .stage2_arm) return error.SkipZigTest; // TODO
    if (builtin.zig_backend == .stage2_spirv64) return error.SkipZigTest;

    const Value = union(enum) {
        values: []const @This(),
        b,

        fn x(value: @This()) !void {
            switch (value.values[0]) {
                .values => return try x(value.values[0]),
                .b => return error.a,
            }
        }
    };
    const a = Value{
        .values = &[1]Value{
            .{
                .values = &[1]Value{.{ .b = {} }},
            },
        },
    };
    try expectError(error.a, Value.x(a));
}

And created the following tests with the following results:

// PASSES (this is pretty much identical to the existing one test)
test "recursive function with inferred error set containing try of itself" {
    const S = struct {
        fn recurseNTimes(n: u8) !void {
            if (n > 10) return error.TooMuchRecursing;
            if (n == 0) return error.DoneRecursing;
            try recurseNTimes(n - 1);
        }
    };
    try expectError(error.DoneRecursing, S.recurseNTimes(5));
    try expectError(error.TooMuchRecursing, S.recurseNTimes(11));
}

// FAILS (should this succeed?)
// ```
//  test/behavior/error.zig:1012:34: error: unable to resolve inferred error set
//        recurseNTimes(n - 1) catch |err| switch (err) {
// ```
test "recursive function explicitly returning errors from switch of itself" {
    const S = struct {
        fn recurseNTimes(n: u8) !void {
            if (n > 10) return error.TooMuchRecursing;
            if (n == 0) return error.DoneRecursing;
            recurseNTimes(n - 1) catch |err| switch (err) {
                error.TooMuchRecursing => unreachable,
                error.DoneRecursing => return error.ReallyDoneRecursing,
                error.ReallyDoneRecursing => return error.ReallyDoneRecursing,
            };
        }
    };
    try expectError(error.ReallyDoneRecursing, S.recurseNTimes(5));
    try expectError(error.TooMuchRecursing, S.recurseNTimes(11));
}

// FAILS (expected since mutual recusive functions are not supported as pointed out by mlugg)
// ```
// test/behavior/error.zig:1045:33: error: unable to resolve inferred error set
//            try recurseNTimesFnA(n - 1);
// ```
test "mutual recursive functions returning inferred error set" {
    const S = struct {
        fn recurseNTimesFnA(n: u8) !void {
            if (n > 10) return error.TooMuchRecursing;
            if (n == 0) return error.DoneRecursing;
            try recurseNTimesFnB(n - 1);
        }
        fn recurseNTimesFnB(n: u8) !void {
            if (n > 10) return error.TooMuchRecursing;
            if (n == 0) return error.DoneRecursing;
            try recurseNTimesFnA(n - 1);
        }
    };

    try expectError(error.DoneRecursing, S.recurseNTimesFnA(5));
    try expectError(error.DoneRecursing, S.recurseNTimesFnB(5));
    try expectError(error.TooMuchRecursing, S.recurseNTimesFnA(11));
    try expectError(error.TooMuchRecursing, S.recurseNTimesFnB(11));
}
  1. Should I open a PR to add these tests?
  2. Should the second test (with the switch case) succeed?

Sign up for free to join this conversation on GitHub. Already have an account? Sign in to comment
Labels
accepted This proposal is planned. contributor friendly This issue is limited in scope and/or knowledge of Zig internals. proposal This issue suggests modifications. If it also has the "accepted" label then it is planned.
Projects
None yet
Development

No branches or pull requests

4 participants