Skip to content

Commit

Permalink
Derive Arbitrary for enums with a single variant
Browse files Browse the repository at this point in the history
Current implementation of deriving macro for arbitrary trait produces
code that fails with a compilation error when invoked on a enum with a
single variant due to type ambiguity in `match` expression scrutinee.
This patch adds special handling of such enums, eliminating the
redundant match and fixing the error.

Resolves #3691
  • Loading branch information
AlgebraicWolf committed Nov 6, 2024
1 parent 4aa739f commit 93788a2
Show file tree
Hide file tree
Showing 3 changed files with 87 additions and 0 deletions.
6 changes: 6 additions & 0 deletions library/kani_macros/src/derive.rs
Original file line number Diff line number Diff line change
Expand Up @@ -364,6 +364,12 @@ fn fn_any_enum(ident: &Ident, data: &DataEnum) -> TokenStream {
quote! {
panic!(#msg)
}
} else if data.variants.len() == 1 {
let variant = data.variants.first().unwrap();
let init = init_symbolic_item(&variant.ident, &variant.fields);
quote! {
#ident::#init
}
} else {
let arms = data.variants.iter().enumerate().map(|(idx, variant)| {
let init = init_symbolic_item(&variant.ident, &variant.fields);
Expand Down
16 changes: 16 additions & 0 deletions tests/ui/derive-arbitrary/single_variant_enum/expected
Original file line number Diff line number Diff line change
@@ -0,0 +1,16 @@
Checking harness check_with_discriminant...
SUCCESS\
"assertion failed: disc == 42"

Checking harness check_with_named_args...
SUCCESS\
"assertion failed: flag as u8 == 0 || flag as u8 == 1"
2 of 2 cover properties satisfied

Checking harness check_with_args...
SUCCESS\
"assertion failed: c <= char::MAX"
2 of 2 cover properties satisfied

Checking harness check_simple...
1 of 1 cover properties satisfied
Original file line number Diff line number Diff line change
@@ -0,0 +1,65 @@
// Copyright Kani Contributors
// SPDX-License-Identifier: Apache-2.0 OR MIT
//! Check that Kani can automatically derive Arbitrary enums.
//! An arbitrary enum should always generate a valid arbitrary variant.
extern crate kani;
use kani::cover;

#[derive(kani::Arbitrary)]
enum Simple {
Empty,
}

#[kani::proof]
fn check_simple() {
match kani::any::<Simple>() {
Simple::Empty => cover!(),
}
}

#[derive(kani::Arbitrary)]
enum WithArgs {
Args(char),
}

#[kani::proof]
fn check_with_args() {
match kani::any::<WithArgs>() {
WithArgs::Args(c) => {
assert!(c <= char::MAX);
cover!(c == 'a');
cover!(c == '1');
}
}
}

#[derive(kani::Arbitrary)]
enum WithNamedArgs {
Args { flag: bool },
}

#[kani::proof]
fn check_with_named_args() {
match kani::any::<WithNamedArgs>() {
WithNamedArgs::Args { flag } => {
cover!(flag as u8 == 0);
cover!(flag as u8 == 1);
assert!(flag as u8 == 0 || flag as u8 == 1);
}
}
}

#[derive(kani::Arbitrary)]
enum WithDiscriminant {
Disc = 42,
}

#[kani::proof]
fn check_with_discriminant() {
let v = kani::any::<WithDiscriminant>();
let disc = v as i8;
match v {
WithDiscriminant::Disc => assert!(disc == 42),
}
}

0 comments on commit 93788a2

Please sign in to comment.