pub struct FSet<T: ?Sized>(/* private fields */);
Expand description
A finite set type usable in pearlite and ghost!
blocks.
If you need an infinite set, see Set
.
§Ghost
Since std::collections::HashSet
and
std::collections::BTreeSet
have finite
capacity, this could cause some issues in ghost code:
ghost! {
let mut set = HashSet::new();
for _ in 0..=usize::MAX as u128 + 1 {
set.insert(0); // cannot fail, since we are in a ghost block
}
proof_assert!(set.len() <= usize::MAX@); // by definition
proof_assert!(set.len() > usize::MAX@); // uh-oh
}
This type is designed for this use-case, with no restriction on the capacity.
Implementations§
Source§impl<T: ?Sized> FSet<T>
impl<T: ?Sized> FSet<T>
Sourcepub fn insert(self, e: T) -> Self
pub fn insert(self, e: T) -> Self
Returns a new set, where e
has been added if it was not present.
logic
Self::add(e, self)
Sourcepub fn remove(self, e: T) -> Self
pub fn remove(self, e: T) -> Self
Returns a new set, where e
is no longer present.
logic
Self::rem(e, self)
Sourcepub fn union(self, other: Self) -> Self
pub fn union(self, other: Self) -> Self
Returns a new set, which is the union of self
and other
.
An element is in the result if it is in self
or if it is in other
.
logic ⚠
Sourcepub fn intersection(self, other: Self) -> Self
pub fn intersection(self, other: Self) -> Self
Returns a new set, which is the union of self
and other
.
An element is in the result if it is in self
or if it is in other
.
logic ⚠
Sourcepub fn difference(self, other: Self) -> Self
pub fn difference(self, other: Self) -> Self
Returns a new set, which is the difference of self
with other
.
An element is in the result if and only if it is in self
but not in other
.
logic ⚠
Sourcepub fn is_subset(self, other: Self) -> bool
pub fn is_subset(self, other: Self) -> bool
Returns true
if every element of self
is in other
.
logic ⚠
Sourcepub fn is_superset(self, other: Self) -> bool
pub fn is_superset(self, other: Self) -> bool
Returns true
if every element of other
is in self
.
logic
Self::is_subset(other, self)
Sourcepub fn disjoint(self, other: Self) -> bool
pub fn disjoint(self, other: Self) -> bool
Returns true
if self
and other
are disjoint.
logic ⚠
Sourcepub fn len(self) -> Int
pub fn len(self) -> Int
Returns the number of elements in the set, also called its length.
logic ⚠
Sourcepub fn peek(self) -> Twhere
T: Sized,
pub fn peek(self) -> Twhere
T: Sized,
Get an arbitrary element of the set.
§Returns
- If the set is nonempty, the result is guaranteed to be in the set
- If the set is empty, the result is unspecified
logic ⚠
Sourcepub fn ext_eq(self, other: Self) -> boolwhere
T: Sized,
pub fn ext_eq(self, other: Self) -> boolwhere
T: Sized,
Extensional equality
Returns true
if self
and other
contain exactly the same elements.
This is in fact equivalent with normal equality.
logic
pearlite! { forall <e: T> self.contains(e) == other.contains(e) }
ensures
result ==> self == other
Source§impl<T> FSet<T>
impl<T> FSet<T>
Sourcepub fn singleton(x: T) -> Self
pub fn singleton(x: T) -> Self
Returns the set containing only x
.
logic
FSet::EMPTY.insert(x)
ensures
forall<y: T> result.contains(y) == (x == y)
Sourcepub fn unions<U>(self, f: Mapping<T, FSet<U>>) -> FSet<U>
pub fn unions<U>(self, f: Mapping<T, FSet<U>>) -> FSet<U>
Returns the union of sets f(t)
over all t: T
.
logic
if self.len() == 0 { FSet::EMPTY } else { let x = self.peek(); f.get(x).union(self.remove(x).unions(f)) }
ensures
forall<y: U> result.contains(y) == exists<x: T> self.contains(x) && f.get(x).contains(y)
Sourcepub fn map<U>(self, f: Mapping<T, U>) -> FSet<U>
pub fn map<U>(self, f: Mapping<T, U>) -> FSet<U>
Returns the image of a set by a function.
logic
FSet::fmap(f, self)
Sourcepub fn filter(self, f: Mapping<T, bool>) -> Self
pub fn filter(self, f: Mapping<T, bool>) -> Self
Returns the subset of elements of self
which satisfy the predicate f
.
logic ⚠
Sourcepub fn cons(s: FSet<T>, ss: FSet<Seq<T>>) -> FSet<Seq<T>>
pub fn cons(s: FSet<T>, ss: FSet<Seq<T>>) -> FSet<Seq<T>>
Returns the set of sequences whose head is in s
and whose tail is in ss
.
logic ⚠
ensures
forall<xs: Seq<T>> result.contains(xs) == (0 < xs.len() && s.contains(xs[0]) && ss.contains(xs.tail()))
Sourcepub fn concat(s: FSet<Seq<T>>, t: FSet<Seq<T>>) -> FSet<Seq<T>>
pub fn concat(s: FSet<Seq<T>>, t: FSet<Seq<T>>) -> FSet<Seq<T>>
Returns the set of concatenations of a sequence in s
and a sequence in t
.
logic ⚠
ensures
forall<xs: Seq<T>> result.contains(xs) == (exists<ys: Seq<T>, zs: Seq<T>> s.contains(ys) && t.contains(zs) && xs == ys.concat(zs))
Sourcepub fn replicate(self, n: Int) -> FSet<Seq<T>>
pub fn replicate(self, n: Int) -> FSet<Seq<T>>
Returns the set of sequences of length n
whose elements are in self
.
logic
pearlite! { if n == 0 { proof_assert! { forall<xs: Seq<T>> xs.len() == 0 ==> xs == Seq::EMPTY }; FSet::singleton(Seq::EMPTY) } else { proof_assert! { forall<xs: Seq<T>, i: Int> 0 < i && i < xs.len() ==> xs[i] == xs.tail()[i-1] }; FSet::cons(self, self.replicate(n - 1)) } }
requires
n >= 0
ensures
forall<xs: Seq<T>> result.contains(xs) == (xs.len() == n && forall<x: T> xs.contains(x) ==> self.contains(x))
Sourcepub fn replicate_up_to(self, n: Int) -> FSet<Seq<T>>
pub fn replicate_up_to(self, n: Int) -> FSet<Seq<T>>
Returns the set of sequences of length at most n
whose elements are in self
.
logic
pearlite! { if n == 0 { proof_assert! { forall<xs: Seq<T>> xs.len() == 0 ==> xs == Seq::EMPTY }; FSet::singleton(Seq::EMPTY) } else { self.replicate_up_to(n - 1).union(self.replicate(n)) } }
requires
n >= 0
ensures
forall<xs: Seq<T>> result.contains(xs) == (xs.len() <= n && forall<x: T> xs.contains(x) ==> self.contains(x))
Source§impl<T: ?Sized> FSet<T>
Ghost definitions
impl<T: ?Sized> FSet<T>
Ghost definitions
Sourcepub fn new() -> Ghost<Self>
pub fn new() -> Ghost<Self>
Create a new, empty set on the ghost heap.
pure
ensures
result.is_empty()
Sourcepub fn len_ghost(&self) -> Int
pub fn len_ghost(&self) -> Int
Returns the number of elements in the set.
If you need to get the length in pearlite, consider using len
.
§Example
use creusot_contracts::{logic::FSet, *};
let mut set = FSet::new();
ghost! {
let len1 = set.len_ghost();
set.insert_ghost(1);
set.insert_ghost(2);
set.insert_ghost(1);
let len2 = set.len_ghost();
proof_assert!(len1 == 0);
proof_assert!(len2 == 2);
};
pure
ensures
result == self.len()
Sourcepub fn contains_ghost(&self, value: &T) -> bool
pub fn contains_ghost(&self, value: &T) -> bool
Returns true if the set contains the specified value.
§Example
use creusot_contracts::{logic::FSet, *};
let mut set = FSet::new();
ghost! {
set.insert_ghost(1);
let (b1, b2) = (set.contains_ghost(&1), set.contains_ghost(&2));
proof_assert!(b1);
proof_assert!(!b2);
};
pure
ensures
result == self.contains(*value)
Sourcepub fn insert_ghost(&mut self, value: T) -> boolwhere
T: Sized,
pub fn insert_ghost(&mut self, value: T) -> boolwhere
T: Sized,
Adds a value to the set.
Returns whether the value was newly inserted. That is:
- If the set did not previously contain this value,
true
is returned. - If the set already contained this value,
false
is returned, and the set is not modified: original value is not replaced, and the value passed as argument is dropped.
§Example
use creusot_contracts::{logic::FSet, *};
let mut set = FSet::new();
ghost! {
let res1 = set.insert_ghost(42);
proof_assert!(res1);
proof_assert!(set.contains(42i32));
let res2 = set.insert_ghost(41);
let res3 = set.insert_ghost(42);
proof_assert!(res2);
proof_assert!(!res3);
proof_assert!(set.len() == 2);
};
pure
ensures
^self == (*self).insert(value)
ensures
result == !(*self).contains(value)
Sourcepub fn insert_ghost_unsized(&mut self, value: Box<T>) -> bool
pub fn insert_ghost_unsized(&mut self, value: Box<T>) -> bool
Same as Self::insert_ghost
, but for unsized values.
pure
ensures
^self == (*self).insert(*value)
ensures
result == !(*self).contains(*value)
Sourcepub fn remove_ghost(&mut self, value: &T) -> bool
pub fn remove_ghost(&mut self, value: &T) -> bool
Removes a value from the set. Returns whether the value was present in the set.
§Example
use creusot_contracts::{logic::FSet, *};
let mut set = FSet::new();
let res = ghost! {
set.insert_ghost(1);
let res1 = set.remove_ghost(&1);
let res2 = set.remove_ghost(&1);
proof_assert!(res1 && !res2);
};
pure
ensures
^self == (*self).remove(*value)
ensures
result == (*self).contains(*value)
Trait Implementations§
impl<T: Clone + Copy> Copy for FSet<T>
Auto Trait Implementations§
impl<T> Freeze for FSet<T>where
T: ?Sized,
impl<T> !RefUnwindSafe for FSet<T>
impl<T> !Send for FSet<T>
impl<T> !Sync for FSet<T>
impl<T> !Unpin for FSet<T>
impl<T> !UnwindSafe for FSet<T>
Blanket Implementations§
Source§impl<T> BorrowMut<T> for Twhere
T: ?Sized,
impl<T> BorrowMut<T> for Twhere
T: ?Sized,
Source§fn borrow_mut(&mut self) -> &mut T
fn borrow_mut(&mut self) -> &mut T
Source§impl<T> CloneToUninit for Twhere
T: Clone,
impl<T> CloneToUninit for Twhere
T: Clone,
Source§impl<T> MakeSized for Twhere
T: ?Sized,
impl<T> MakeSized for Twhere
T: ?Sized,
Source§fn make_sized(&self) -> Box<T>
fn make_sized(&self) -> Box<T>
logic ⚠
ensures
*result == *self