#![feature(coverage_attribute)]
//@ edition: 2021
//@ compile-flags: -Zcoverage-options=branch
//@ llvm-cov-flags: --show-branches=count

// Tests for branch coverage of the lazy boolean operators `&&` and `||`,
// as ordinary expressions that aren't part of an `if` condition or similar.

use core::hint::black_box;

// Helper macro to prevent start-of-function spans from being merged into
// spans on the lines we care about.
macro_rules! no_merge {
    () => {
        for _ in 0..1 {}
    };
}

fn branch_and(a: bool, b: bool) {
    no_merge!();

    //      |13  |18 (no branch)
    let c = a && b;
    black_box(c);
}

fn branch_or(a: bool, b: bool) {
    no_merge!();

    //      |13  |18 (no branch)
    let c = a || b;
    black_box(c);
}

// Test for chaining one operator several times.
fn chain(x: u32) {
    no_merge!();

    //      |13      |22      |31      |40 (no branch)
    let c = x > 1 && x > 2 && x > 4 && x > 8;
    black_box(c);

    //      |13      |22      |31      |40 (no branch)
    let d = x < 1 || x < 2 || x < 4 || x < 8;
    black_box(d);
}

// Test for nested combinations of different operators.
fn nested_mixed(x: u32) {
    no_merge!();

    //       |14      |23         |35      |44 (no branch)
    let c = (x < 4 || x >= 9) && (x < 2 || x >= 10);
    black_box(c);

    //       |14      |23        |34       |44 (no branch)
    let d = (x < 4 && x < 1) || (x >= 8 && x >= 10);
    black_box(d);
}

#[coverage(off)]
fn main() {
    // Use each set of arguments (2^n) times, so that each combination has a
    // unique sum, and we can use those sums to verify expected control flow.
    // 1x (false, false)
    // 2x (false, true)
    // 4x (true, false)
    // 8x (true, true)
    for a in [false, true, true, true, true] {
        for b in [false, true, true] {
            branch_and(a, b);
            branch_or(a, b);
        }
    }

    for x in 0..16 {
        chain(x);
        nested_mixed(x);
    }
}