use super::super::prelude::*;
verus! {
#[verifier::external_trait_specification]
pub trait ExIndex<Idx> where Idx: ?Sized {
type Output: ?Sized;
type ExternalTraitSpecificationFor: core::ops::Index<Idx>;
}
#[verifier::external_trait_specification]
pub trait ExIndexMut<Idx>: core::ops::Index<Idx> where Idx: ?Sized {
type ExternalTraitSpecificationFor: core::ops::IndexMut<Idx>;
}
#[verifier::external_trait_specification]
pub trait ExInteger: Copy {
type ExternalTraitSpecificationFor: Integer;
}
#[verifier::external_trait_specification]
pub trait ExSpecOrd<Rhs> {
type ExternalTraitSpecificationFor: SpecOrd<Rhs>;
}
#[verifier::external_trait_specification]
pub trait ExAllocator {
type ExternalTraitSpecificationFor: core::alloc::Allocator;
}
#[verifier::external_trait_specification]
pub trait ExFreeze {
type ExternalTraitSpecificationFor: core::marker::Freeze;
}
#[verifier::external_trait_specification]
pub trait ExDebug {
type ExternalTraitSpecificationFor: core::fmt::Debug;
}
#[verifier::external_trait_specification]
pub trait ExFrom<T>: Sized {
type ExternalTraitSpecificationFor: core::convert::From<T>;
fn from(v: T) -> (ret: Self);
}
#[verifier::external_trait_specification]
pub trait ExInto<T>: Sized {
type ExternalTraitSpecificationFor: core::convert::Into<T>;
fn into(self) -> (ret: T);
}
pub assume_specification<T, U: From<T>>[ T::into ](a: T) -> (ret: U)
ensures
call_ensures(U::from, (a,), ret),
;
#[verifier::external_trait_specification]
pub trait ExPartialEq<Rhs: ?Sized> {
type ExternalTraitSpecificationFor: core::cmp::PartialEq<Rhs>;
}
#[verifier::external_trait_specification]
pub trait ExEq: PartialEq {
type ExternalTraitSpecificationFor: core::cmp::Eq;
}
#[verifier::external_trait_specification]
pub trait ExPartialOrd<Rhs: ?Sized>: PartialEq<Rhs> {
type ExternalTraitSpecificationFor: core::cmp::PartialOrd<Rhs>;
}
#[verifier::external_trait_specification]
pub trait ExOrd: Eq + PartialOrd {
type ExternalTraitSpecificationFor: Ord;
}
#[verifier::external_trait_specification]
pub trait ExHash {
type ExternalTraitSpecificationFor: core::hash::Hash;
}
#[verifier::external_trait_specification]
pub trait ExPtrPointee {
type ExternalTraitSpecificationFor: core::ptr::Pointee;
type Metadata:
Copy + Send + Sync + Ord + core::hash::Hash + Unpin + core::fmt::Debug + Sized + core::marker::Freeze;
}
#[verifier::external_trait_specification]
pub trait ExIterator {
type ExternalTraitSpecificationFor: core::iter::Iterator;
type Item;
fn next(&mut self) -> Option<Self::Item>;
}
#[verifier::external_trait_specification]
pub trait ExIntoIterator {
type ExternalTraitSpecificationFor: core::iter::IntoIterator;
}
#[verifier::external_trait_specification]
pub trait ExIterStep: Clone + PartialOrd + Sized {
type ExternalTraitSpecificationFor: core::iter::Step;
}
#[verifier::external_trait_specification]
pub trait ExBorrow<Borrowed> where Borrowed: ?Sized {
type ExternalTraitSpecificationFor: core::borrow::Borrow<Borrowed>;
}
#[verifier::external_trait_specification]
pub trait ExStructural {
type ExternalTraitSpecificationFor: Structural;
}
pub assume_specification<T>[ core::mem::swap::<T> ](a: &mut T, b: &mut T)
ensures
*a == *old(b),
*b == *old(a),
opens_invariants none
no_unwind
;
#[verifier::external_type_specification]
#[verifier::accept_recursive_types(V)]
#[verifier::ext_equal]
pub struct ExOption<V>(core::option::Option<V>);
#[verifier::external_type_specification]
#[verifier::accept_recursive_types(T)]
#[verifier::reject_recursive_types_in_ground_variants(E)]
pub struct ExResult<T, E>(core::result::Result<T, E>);
pub open spec fn iter_into_iter_spec<I: Iterator>(i: I) -> I {
i
}
#[verifier::when_used_as_spec(iter_into_iter_spec)]
pub assume_specification<I: Iterator>[ I::into_iter ](i: I) -> (r: I)
ensures
r == i,
;
#[verifier::external_type_specification]
#[verifier::external_body]
pub struct ExDuration(core::time::Duration);
#[verifier::external_type_specification]
#[verifier::accept_recursive_types(V)]
pub struct ExPhantomData<V: ?Sized>(core::marker::PhantomData<V>);
pub assume_specification[ core::intrinsics::likely ](b: bool) -> (c: bool)
ensures
c == b,
;
pub assume_specification[ core::intrinsics::unlikely ](b: bool) -> (c: bool)
ensures
c == b,
;
pub assume_specification<T, F: FnOnce() -> T>[ bool::then ](b: bool, f: F) -> (ret: Option<T>)
ensures
if b {
ret.is_some() && f.ensures((), ret.unwrap())
} else {
ret.is_none()
},
;
#[verifier::external_type_specification]
#[verifier::external_body]
#[verifier::reject_recursive_types_in_ground_variants(V)]
pub struct ExManuallyDrop<V: ?Sized>(core::mem::ManuallyDrop<V>);
pub(crate) trait TrustedSpecSealed {
}
#[allow(private_bounds)]
pub trait IndexSetTrustedSpec<Idx>: core::ops::IndexMut<Idx> + TrustedSpecSealed {
spec fn spec_index_set_requires(&self, index: Idx) -> bool;
spec fn spec_index_set_ensures(
&self,
new_container: &Self,
index: Idx,
val: Self::Output,
) -> bool where Self::Output: Sized;
}
#[verifier(external_body)]
pub fn index_set<T, Idx, E>(container: &mut T, index: Idx, val: E) where
T: ?Sized + core::ops::IndexMut<Idx> + core::ops::Index<Idx, Output = E> + IndexSetTrustedSpec<
Idx,
>,
requires
old(container).spec_index_set_requires(index),
ensures
old(container).spec_index_set_ensures(container, index, val),
{
container[index] = val;
}
impl<T, const N: usize> TrustedSpecSealed for [T; N] {
}
impl<T, const N: usize> IndexSetTrustedSpec<usize> for [T; N] {
open spec fn spec_index_set_requires(&self, index: usize) -> bool {
0 <= index < N
}
open spec fn spec_index_set_ensures(&self, new_container: &Self, index: usize, val: T) -> bool {
new_container@ === self@.update(index as int, val)
}
}
impl<T> TrustedSpecSealed for [T] {
}
impl<T> IndexSetTrustedSpec<usize> for [T] {
open spec fn spec_index_set_requires(&self, index: usize) -> bool {
0 <= index < self@.len()
}
open spec fn spec_index_set_ensures(&self, new_container: &Self, index: usize, val: T) -> bool {
new_container@ == self@.update(index as int, val)
}
}
pub assume_specification[ core::hint::unreachable_unchecked ]() -> !
requires
false,
;
} macro_rules! impl_from_spec {
($from: ty => [$($to: ty)*]) => {
verus!{
$(
pub assume_specification[ <$to as core::convert::From<$from>>::from ](a: $from) -> (ret: $to)
ensures
ret == a as $to,
;
)*
}
};
}
impl_from_spec! {u8 => [u16 u32 u64 usize u128]}
impl_from_spec! {u16 => [u32 u64 usize u128]}
impl_from_spec! {u32 => [u64 u128]}
impl_from_spec! {u64 => [u128]}